初始化顺序的规则
1.在一个类的对象实例化时,成员变量首先初始化,然后才调用构造器,无论书写顺序。如果调用构造器前,没有显式初始化,那么会赋默认值。
这样做法的原因可以理解为:构造器执行时可能会用到一些成员变量的初值。
2.static变量早于所有其他的类成员变量初始化,同样无论书写顺序。但是static变量仅在所在类第一次被使用时初始化一次。
3.基类构造器总是在导出类的构造过程中被调用,而且按照继承层级逐渐向上链接(调用顺序则是从基类开始向下)。可以理解为,这么做的逻辑关系是在一个类构建时可能会用到其父类的成员、方法。在清理时顺序相反。
4.成员的初始化方法(包括基本数据类型的赋值)在基类构造器调用之后才会被调用。最初时,分配给对象的存储空间初始化二进制的零。
例一出自《Java编程思想》第5.7.2节,为了便于演示初始化顺序,进行了缩减和重新编号。用构造器的参数标明执行顺序,演示1~2条规则:
class Bowl {
Bowl(int marker) {
System.out.println("Bowl(" + marker + ")");
}
}
class Cupboard {
Bowl bowl1 = new Bowl(3);
static Bowl bowl2 = new Bowl(1);
int i;
static int j = 5;
Cupboard() {
System.out.println("i:" + i);
bowl4 = new Bowl(j);
j = 6;
}
Bowl bowl3 = new Bowl(4);
static Bowl bowl4 = new Bowl(2);
}
public class ParaInitialization {
public static void main(String args[]) {
new Cupboard();
new Cupboard();
}
}
输出及对应注释:
Bowl(1) //第一个static变量
Bowl(2) //第二个static变量
Bowl(3) //第一个对象的第一个非static成员变量
Bowl(4) //第一个对象的第一个非static成员变量
i:0 //未显示初始化的成员变量
Bowl(5) //更改static变量的值
Bowl(3) //第二个对象的第一个非static成员变量
Bowl(4) //第二个对象的第二个非static成员变量
i:0
Bowl(6)
例二是一个演示第3条规则的简单示例。
class A {
A() {
System.out.println("A");
}
}
class B extends A {
B() {
System.out.println("B");
}
}
class C extends B {
C() {
System.out.println("C");
}
}
public class hrt {
public static void main(String args[]) {
new C();
}
}
输出
A
B
C
例三用于演示规则4。调用父类构造器时,构造器中的方法被子类方法覆盖。
class Glyph {
void draw() {
System.out.println("Glyph.draw(");
}
Glyph() {
System.out.println("Glyph() before draw()");
draw();
System.out.println("Glyph() after draw()");
}
}
class RoundGlyph extends Glyph {
int radius = 1;
RoundGlyph(int r) {
radius = r;
System.out.println("RoundGlyph.RoundGlyph(), radius = " + radius);
}
void draw() {
System.out.println("RoundGlyph.draw(), radius = " + radius);
}
}
public class PolyConstructors {
public static void main(String[] args) {
new RoundGlyph(5);
}
}
这么多条规则,记起来实在让人头大。将它们按顺序编排会易读很多。
对象初始化顺序,如果有对应成员/父类的才执行对应条目:
1.将分配给对象的存储空间初始化为二进制的零;
2.调用基类构造器,从最顶层/根的基类开始;
3.按照声明的顺序,使用直接的赋值或者初始化方法,先依次初始化static变量,再依次初始化非static变量;
4.调用本对象所属类的构造器。