从外部看来,派生类是一个与基类具有相同接口的新类,或许还会有一些额外的的方法和域 。但继承并不仅仅是类的复用。当创建了一个派生类的对象时,该类包含了一个基类的子对象。这个子对象和你用基类直接创建的对象没有什么两样。二者的区别在于,后者来自于外部,而基类的子对象来自于派生类对象的内部。对基类的子对象初始化时至关重要的,而且也只有一种方法来保证这一点,那就是在派生类的构造器中调用基类的构造器,而基类的构造器具有执行基类初始化所需的所有能力和知识。

java会自动在派生类的构造器中插入对基类的构造器的调用。下例展示了上述机制:

public class B1 { B1() { //super(); System.out.println("B1()"); } } public class B2 extends B1 { B2() { //super(); System.out.println("B2()"); } } public class B3 extends B2 { B3() { //super(); System.out.println("B3()"); } public static void main(String[] args) { B3 b3 = new B3(); System.out.println(); } } 输出的结果为: B1() B2() B3()

从上述代码中可以看到,构建过程是从基类"向外"扩散的,所以在派生类的构造器在没有访问之前基类的构造器已经被访问了。即使你不为B3创建构造器,编译器也会为你提供一个默认的构造器去调用基类的构造器。将上诉代码中的三个super();

去除注释结果是一样的,只不过是我现实的调用了基类的构造器。

带参数的构造器

上诉事例中构造器都是无参的,编译器可以轻松的调用它因为不必考虑要传递什么样的参数。但是如果没有默认基类构造器,或者想调用一个带参数的基类构造器,那就必须使用关键字super现实地编写调用基类构造器的代码,并且匹配适当的参数列表。

public class B1 { B1(int i) { System.out.println("B1()"); } } public class B2 extends B1 { B2(int i) { super(i); System.out.println("B2()"); } } public class B3 extends B2 { B3() { super(11); System.out.println("B3()"); } public static void main(String[] args) { B3 b3 = new B3(); System.out.println(); } }

如果没有在B3中调用基类的构造器,那么编译器就会提示错误:"未定义隐式超构造函数 B2()。必须显式调用另一个构造函数"。而且,调用基类的构造器必须是你在派生类构造器中做的第一件事(调用基类构造器的代码必须放在派生类构造器的第一行)。

总结:当创建一个派生类的对象时,先被访问的是基类的构造器,然后是派生类的构造器。如果基类没有提供默认的构造器,那么派生类就要用关键字super显示地调用基类的那个构造器。