子类构造方法显示调用父类构造方法并传递子类成员变量,编译器报错并提示将传递的成员变量用static修饰。错误原因显示调用构造函数时,不能引用实例字段。涉及到对象声明和创建时,类加载时类属性初始化、父子类对象属性的初始化及父子类的构造方法执行的顺序。
由static修饰的属性和方法所属为其所在类,在内存中只有一份,为类所创建的所有对象共享,随类的加载先对static修饰内容初始化,继承中先初始父类的static,之后初始子类static内容。
调用父类构造方法虽然在子类构造方法第一行,但执行时更靠前,如果加上构造代码块进行比较,构造代码块执行在构造方法之前,父子类的构造代码块和构造方法的执行顺序应为 构造代码块(父类)-> 构造方法(父类) -> 构造代码块(子类)-> 构造方法(子类)。成员变量的初始化在各自构造代码块之前。
最终执行顺序为:static(父类)-> static(子类)-> 成员变量初始化(父类)-> 构造代码块(父类)-> 构造方法(父类) -> 成员变量初始化(子类)-> 构造代码块(子类)-> 构造方法(子类)。
显示调用构造函数时,不能引用实例字段。实例字段初始化之后才可使用,是在父类构造方法执行时,子类成员变量未被初始化,而静态修饰的子类属性初始化是先于父类构造方法执行。
测试:
class Fu {
static {
System.out.println("静态代码块Fu"); //1
}
{
System.out.println("构造代码块Fu"); //3
}
public Fu() {
System.out.println("构造方法Fu"); //4
}
}
class Zi extends Fu {
static {
System.out.println("静态代码块Zi"); //2
}
{
System.out.println("构造代码块Zi"); //5
}
public Zi() {
System.out.println("构造方法Zi"); //6
}
}
测试结果:
不包含成员变量的测试,可在个各个代码块输出成员变量来测试;
JVM中应该由更详细的解释,这里只是简单测试得出的测试结果,如果有更好理解,可以私信或评论交流。