对super()的调用必须是构造函数体中的第一个语句

mat*_*teo 7 java constructor super

我正在编写一个LoginRequest类的构造函数,它扩展了一个名为JsobObjectRequest的类(来自Android中的Volley框架,但这与问题完全无关)

使用此代码:

 public LoginRequest(String username, String password, Response.Listener<JSONObject> responseListener, Response.ErrorListener errorListener) {
        Boolean hasCredentials=(username!=null && password!=null);
        int method=hasCredentials? Method.POST:Request.Method.GET;
        super(method, API_URL_LOGIN, null, responseListener, errorListener);

        this.username=username;
        this.password=password;

    }
Run Code Online (Sandbox Code Playgroud)

我得到错误:调用super()必须是构造函数体中的第一个语句

相反,这段代码编译得很好:

 public LoginRequest(String username, String password, Response.Listener<JSONObject> responseListener, Response.ErrorListener errorListener) {
        super((username!=null && password!=null)? Method.POST:Request.Method.GET, API_URL_LOGIN, null, responseListener, errorListener);

        this.username=username;
        this.password=password;

    }
Run Code Online (Sandbox Code Playgroud)

但这不是有效的完全相同的事情吗?在这两种情况下,在调用超级构造函数之前,根据传递给子类构造函数的参数值,进行一些简单的计算.为什么编译器不能编译第一个例子,因为它可以编译第二个例子?

call-super-constructor-must-first-statement语句规范是否比它需要的更简单,或者我错过了什么?

编辑:这被错误地标记为重复为什么this()和super()必须是构造函数中的第一个语句?.这个问题更通用,并询问为什么super()必须是第一个声明.这里的问题是为什么像我发布的那样的案例会破坏这些要求(并且在这个问题中得到了令人满意的回答)

rge*_*man 11

Java强制对super(显式或非)的调用必须是构造函数中的第一个语句.这是为了防止在初始化对象的超类部分之前初始化对象的子类部分.

在你的情况下,除了本地的"琐碎计算"之外你什么都不做,所以考虑到所有事情,它都没关系.但是,Java的编译器没有进入确定调用之前是否super实际执行任何初始化的语句的级别.它只是不允许以前的所有陈述super().

  • 我明白了,谢谢.令人惊讶的是,编译器负责检测无法访问的代码,但无法检查构造函数中的代码是否可以执行实际的初始化. (3认同)
  • 它能够检查这样的事情.但是,在编译期间可能需要花费太多时间(猜测). (2认同)