首先看一段代码来考考大家,请大家在不运行程序的情况下,输出以下代码的结果:
class Insect { int i = 9; int j; Insect() { prt("i = " + i + ", j = " + j); j = 39; } static int x1 = prt("static Insect.x1 initialized"); static int prt(String s) { System.out.println(s); return 47; } } public class Test extends Insect { int k = prt("Beetle.k initialized"); Test() { prt("k = " + k); prt("j = " + j); } static int x2 = prt("static Beetle.x2 initialized"); static int prt(String s) { System.out.println(s); return 63; } public static void main(String[] args) { prt("Beetle constructor"); Test b = new Test(); } }
代码的结果:
static Insect.x1 initialized static Beetle.x2 initialized Beetle constructor i = 9, j = 0 Beetle.k initialized k = 63 j = 39
对Test运行Java 时,发生的第一件事情是装载程序到外面找到那个类。在装载过程中,装载程序注意它
有一个基础类(即extends 关键字要表达的意思),所以随之将其载入。无论是否准备生成那个基础类的一
个对象,这个过程都会发生(请试着将对象的创建代码当作注释标注出来,自己去证实)。
若基础类含有另一个基础类,则另一个基础类随即也会载入,以此类推。接下来,会在根基础类(此时是
Insect)执行 static 初始化,再在下一个衍生类执行,以此类推。保证这个顺序是非常关键的,因为衍生类
的初始化可能要依赖于对基础类成员的正确初始化。
此时,必要的类已全部装载完毕,所以能够创建对象。首先,这个对象中的所有基本数据类型都会设成它们
的默认值,而将对象句柄设为null。随后会调用基础类构建器。在这种情况下,调用是自动进行的。但也完
全可以用super 来自行指定构建器调用(就象在Test()构建器中的第一个操作一样)。基础类的构建采用
与衍生类构建器完全相同的处理过程。基础顺构建器完成以后,实例变量会按本来的顺序得以初始化。最
后,执行构建器剩余的主体部分。
补充:
class Father{ Father(){ System.out.println("father run!"); } } class Son extends Father{ Son(){ System.out.println("son run!"); } } public class Demo { public static void main(String[] args) { new Son(); //father run! //son run! } }
在子类构造对象时,访问子类构造函数时,父类也运行。为什么哪?
在子类构造函数中第一行有一个默认的隐式语句:super();并且子类实例化过程中,子类所有的构造函数默认都会访问父类中的空参数的构造函数,如果父类中没有定义空参的构造函数,那么子类构造函数必须用super显式调用父类中的哪个构造函数。
Son(){
super(4);
System.out.println("son run!");
}
为什么子类实例化过程中要调用父类的构造函数?
class Father{ int num; Father(){ num=10; } } public class Son extends Father{ Son(){ System.out.println(num); } public static void main(String[] args) { new Son();//10 } }那是因为子类继承了父类,获取到了父类中的内容(属性),所以在使用父类内容之前,要先看父类是如何对自己的内容进行初始化的。Super语句要定义在子类构造函数的第一行,因为父类的初始化动作要先完成。
class Father{ int num; Father(){ } } public class Son extends Father{ Son(){//这个调用父类构造函数 } Son(int num){ this();//此时有this情形 } public static void main(String[] args) { new Son(20);//20 } }
class Father{ int num=10; Father(){ show(); } public void show(){ System.out.println("Father run "+num); } } public class Son extends Father{ int num=20; Son(){ show(); } public void show(){ System.out.println("son run "+num); } public static void main(String[] args) { new Son(); } } //son run 0 //son run 20一个对象实例化过程Person p=new Person():