子类继承父类,创建子类对象时,先调用父类的构造函数,再调用自己的构造函数,如果父类的构造函数中调用了一个方法,且方法被子类覆写,那么就会调用子类的覆写的方法
举例:
package constructor; class A { A() { draw(); System.out.println("A is constructing"); } void draw() { System.out.println("A is drawing!"); } } public class B extends A { private int m = 1; B(int m) { this.m = m; System.out.println("B is constructing! m = "+Integer.toString(m)); } void draw() { System.out.println("B is drawing! m = "+Integer.toString(m)); } public static void main(String[] args) { new B(10); //输出如下 /* B is drawing! m = 0 A is constructing B is constructing! m = 10 */ } }如上所示:
你会发现m的值第一次应该为1,却输出了0,那是因为构造对象时,对象的存储空间被初始化为二进制的0。
创建子类对象 B 时,初始化的实际过程如下
(1)在任何其他事物发生之前,将分配给对象的存储空间初始化为二进制的0
(2)调用父类的构造器,调用相应的方法draw,由于(1)的缘故,导致m = 0
(3)按类成员进行初始化,此时m = 1
(4)调用子类的构造器
因此,一般在构造器中调用final方法或private方法(private方法自动属于final方法),这样子类就不会覆写父类的这些final方法,从而就无法在调用父类构造器的时候调用子类的覆写的方法,从而避免了m = 0情况,因为这样就不会调用子类的draw方法