我今天早些时候在代码中结束了以下场景(我承认有点奇怪,我已经重构了).当我运行单元测试时,我发现在超类构造函数运行的时候没有设置字段初始化.我意识到我并不完全理解构造函数/字段初始化的顺序,所以我希望有人向我解释这些发生的顺序.
class Foo extends FooBase {
String foo = "foobar";
@Override
public void setup() {
if (foo == null) {
throw new RuntimeException("foo is null");
}
super.setup();
}
}
class FooBase {
public FooBase() {
setup();
}
public void setup() {
}
}
@Test
public void testFoo() {
new Foo();
}
来自JUnit的缩写回溯如下,我想我期望$Foo.< init>设置foo.
$Foo.setup $FooBase.<init> $Foo.<init> .testFoo
解决方法
是的,在Java中(例如,与C#不同)字段初始值设定项在超类构造函数之后调用.这意味着在执行字段初始化程序之前将调用构造函数的任何重写方法调用.
订购是:
>初始化超类(递归调用这些步骤)
>执行字段初始值设定项
>执行构造函数体(在任何构造函数链接之后,已在步骤1中发生)
基本上,在构造函数中调用非final方法是个坏主意.如果您打算这样做,请将其清楚地记录下来,以便任何覆盖该方法的人都知道在执行字段初始化程序(或构造函数体)之前将调用该方法.
有关详细信息,请参见JLS section 12.5.