当前位置: 首页 > 知识库问答 >
问题:

为什么this()和super()必须是构造函数中的第一条语句?

秦宏盛
2023-03-14

Java要求,如果您在构造函数中调用this()或super(),它必须是第一条语句。为什么?

例如:

public class MyClass {
    public MyClass(int x) {}
}

public class MySubClass extends MyClass {
    public MySubClass(int a, int b) {
        int c = a + b;
        super(c);  // COMPILE ERROR
    }
}

Sun编译器说“调用super必须是构造函数中的第一条语句”。Eclipse编译器说“构造函数调用必须是构造函数中的第一条语句”。

但是,您可以通过稍微重新排列代码来解决这个问题:

public class MySubClass extends MyClass {
    public MySubClass(int a, int b) {
        super(a + b);  // OK
    }
}

下面是另一个例子:

public class MyClass {
    public MyClass(List list) {}
}

html" target="_blank">public class MySubClassA extends MyClass {
    public MySubClassA(Object item) {
        // Create a list that contains the item, and pass the list to super
        List list = new ArrayList();
        list.add(item);
        super(list);  // COMPILE ERROR
    }
}

public class MySubClassB extends MyClass {
    public MySubClassB(Object item) {
        // Create a list that contains the item, and pass the list to super
        super(Arrays.asList(new Object[] { item }));  // OK
    }
}

因此,它不会阻止您在调用super之前执行逻辑。它只是阻止您执行无法容纳在单个表达式中的逻辑。

调用this()也有类似的规则。编译器会说“call to this必须是constructor中的first statement”。

为什么编译器会有这些限制呢?你能给出一个代码示例,如果编译器没有这个限制,那么会发生一些不好的事情吗?

共有2个答案

莫欣悦
2023-03-14

通过链接构造函数和静态方法,我找到了一种解决这一问题的方法。我想做的事情是这样的:

public class Foo extends Baz {
  private final Bar myBar;

  public Foo(String arg1, String arg2) {
    // ...
    // ... Some other stuff needed to construct a 'Bar'...
    // ...
    final Bar b = new Bar(arg1, arg2);
    super(b.baz()):
    myBar = b;
  }
}

因此,基本上构造一个基于构造函数参数的对象,将该对象存储在一个成员中,并将该对象上的方法的结果传递到super的构造函数中。使成员final也是相当重要的,因为类的性质是它是不可变的。请注意,构造Bar实际上需要一些中间对象,因此在我的实际用例中,它不能简化为一行程序。

最后我做了这样的工作:

public class Foo extends Baz {
  private final Bar myBar;

  private static Bar makeBar(String arg1,  String arg2) {
    // My more complicated setup routine to actually make 'Bar' goes here...
    return new Bar(arg1, arg2);
  }

  public Foo(String arg1, String arg2) {
    this(makeBar(arg1, arg2));
  }

  private Foo(Bar bar) {
    super(bar.baz());
    myBar = bar;
  }
}

合法代码,并在调用超级构造函数之前完成执行多条语句的任务。

班承德
2023-03-14

需要在子类“构造器”之前调用父类“构造器”。这将确保如果在构造函数中调用父类上的任何方法,则父类已经正确设置。

您要做的,将参数传递给超级构造函数是完全合法的,您只需要像现在这样内联构造这些参数,或者将它们传递给您的构造函数,然后将它们传递给super:

public MySubClassB extends MyClass {
        public MySubClassB(Object[] myArray) {
                super(myArray);
        }
}

如果编译器没有强制执行此操作,则可以执行以下操作:

public MySubClassB extends MyClass {
        public MySubClassB(Object[] myArray) {
                someMethodOnSuper(); //ERROR super not yet constructed
                super(myArray);
        }
}

类具有默认构造函数的情况下,编译器会自动为您插入对super的调用。由于Java中的每个类都继承自object,因此必须以某种方式调用objects构造函数并首先执行它。编译器自动插入super()允许这样做。强制super首先出现,强制构造函数主体以正确的顺序执行,该顺序为:对象->父级->子级->子级->子级->子级->次级

 类似资料:
  • 问题内容: Java要求,如果你在构造函数中调用this()或super(),则它必须是第一条语句。为什么? 例如: Sun编译器说“对super的调用必须是构造函数中的第一条语句”。Eclipse编译器说“构造函数调用必须是构造函数中的第一条语句”。 但是,你可以通过重新安排一些代码来解决此问题: 这是另一个示例: 因此,这不会阻止你在调用super之前执行逻辑。这只是阻止你执行无法包含在单个表

  • 这个Super()的替代品是什么;我代码中的语句...因为它向我显示了一个名为:构造函数调用必须是构造函数中的第一个语句的错误。

  • 问题内容: 我正在编写一个LoginRequest类的构造函数,该类扩展了一个名为JsobObjectRequest的类(来自Android的Volley框架,但这与问题完全无关) 使用此代码: 我收到错误:对super()的调用必须是构造函数主体中的第一条语句 相反,此代码可以正常编译: 但这实际上不是一回事吗?在这两种情况下,根据传递给子类构造函数的参数值,在调用超级构造函数之前都要进行一些简

  • 我正在编写一个LoginRequest类的构造函数,该类扩展了一个名为JsobObjectRequest的类(来自Android的凌空框架,但这与问题完全无关) 使用此代码: 我得到了一个错误:对super()的调用必须是构造函数体中的第一条语句 相反,这段代码编译得很好: 但事实上不是完全一样吗?在这两种情况下,在调用超级构造函数之前,都会根据传递给子类构造函数的参数值进行一些简单的计算。既然编

  • 无法理解为什么会出现此错误。在记事本中工作,通过cmd运行javac(显然更多的是文件)。如果你需要我链接更多,我可以。我的错误发生在“setCruiseShip”中,我试图将电流链接到超类。错误:对super的调用必须是构造函数中的第一条语句。

  • 如果将移到构造函数的最后一行,我不明白为什么下面的代码会显示错误。 我已经检查了很多关于StackOverflow的答案,但我仍然不能理解这其中的原因。请帮我用一些简单的例子和解释弄清楚这个错误。