在子类构造函数中,发现,访问子类构造函数时,父类也运行了。原因是什么呢?
在子类的构造函数里第一行有一个默认的隐式语句:super()
ExtendsDemo.java
class Fu
{
Fu()
{
System.out.println("fu run");
}
}
class Zi extends Fu
{
Zi()
{
//super(); //调用的是父类中的空参数的构造函数
System.out.println("zu run");
}
}
class ExtendsDemo
{
public static void main(String[] args)
{
new Zi();
}
}
输出:
我们也可访问父类中带有参数的构造函数:
class Fu
{
Fu(int x)
{
System.out.println("fu run");
}
}
class Zi extends Fu
{
Zi()
{
super(4); //父类有带参数的构造函数
System.out.println("zu run");
}
}
子类的实例化过程:
子类中所有的构造函数默认都会访问父类中的空参数的构造函数。当然,如果子类中指定了访问父类带参数的构造函数,就不会访问父类默认的构造函数(好像是废话哈~~)
这就意味着如果父类中没有默认的构造函数,子类尝试调用父类的默认构造函数,程序就会报错:
class Fu
{
Fu(int x) //指定了新的构造函数,默认的构造函数就没有了
{
System.out.println("fu run 2");
}
}
class Zi extends Fu
{
Zi()
{
super(4); //父类有带参数的构造函数
System.out.println("zu run 1");
}
Zi(int x)
{
//super(); //默认会访问父类的构造函数
System.out.println("zu run 2");
}
}
class ExtendsDemo
{
public static void main(String[] args)
{
new Zi(6);
}
}
输出:
所以这时候就需要在子类中指定调用父类带参数的构造函数:
class Fu
{
Fu(int x) //指定了新的构造函数,默认的构造函数就没有了
{
System.out.println("fu run 2");
}
}
class Zi extends Fu
{
Zi()
{
super(4); //父类有带参数的构造函数
System.out.println("zu run 1");
}
Zi(int x)
{
super(x); //默认会访问父类的构造函数
System.out.println("zu run 2");
}
}
class ExtendsDemo
{
public static void main(String[] args)
{
new Zi(6);
}
}
输出:
为什么子类默认会访问父类的默认构造函数呢?
那是因为:子类继承了父类,获取到了父类中内容(属性),所以在使用父类内容之前,要先看父类是如何对自己的内容进行初始化的,所以子类在构造对象时候,必须访问父类的构造函数,为了完成这个必须的动作,就在子类的构造函数中加入了super()语句、
如果父类中没有定义空参数构造函数,那么子类的构造函数必须用super明确要调用父类中哪个构造函数,否则子类无法完成初始化。
注意:super语句必须要定义在子类构造函数的第一行,因为父类的初始化动作要先完成。
//----------------------------------------------------------------------------------------------------------------------------------------------------------------
同时子类构造函数如果使用this调用了本类构造函数时,那么super就没有了,因为super和this都只能定义在第一行,所以只能有一个,但是可以保证的是,子类中肯定会有其他的构造函数访问父类的构造函数。
class Fu
{
Fu(int x) //指定了新的构造函数,默认的构造函数就没有了
{
System.out.println("fu run 2");
}
}
class Zi extends Fu
{
Zi()
{
super(4); //父类有带参数的构造函数
System.out.println("zu run 1");
}
Zi(int x)
{
this();
//super(x); //默认会访问父类的构造函数
System.out.println("zu run 2");
}
}
class ExtendsDemo
{
public static void main(String[] args)
{
new Zi(6);
}
}
输出:
注意的问题:
java中任何类默认会继承一个根类——Object,主动继承这个类或者不继承这个类写法都可以。
class Fu { Fu() { super(); show(); return; } void show() { System.out.println("fu show"); } } class Zi extends Fu { int num = 8; Zi() { super(); System.out.println("zi cons num..."+num); } void show() { System.out.println("zi show..."+num); } } class Demo { public static void main(String[] args) { Zi z = new Zi(); z.show(); } }
输出:
对象实例化图解:
通过super初始化父类内容时,子类的成员变量并未显示初始化,等super()父类初始化完毕后,才进行子类的成员变量显式初始化。
对象的实例化过程步骤总结:
Person p = new Person();
1、JVM会读取指定路径下的Person.class文件,并加载进内存。并会先加载Person的父类(如果有直接的父类的情况下)
2、在堆内存中开辟空间,分配地址。
3、并在对象空间中,对对象中的属性进行默认初始化。(不是显式初始化)
4、调用对应的构造函数进行初始化。
5、在构造函数中,第一行会先调用父类的构造函数进行初始化。
6、父类初始化完毕后,在对子类的属性进行显式初始化。
7、再进行子类构造函数的特定初始化。
8、初始化完毕够,将地址值赋值给引用变量。