Java语言深入-构造函数的继承问题

发表于:2007-06-22来源:作者:点击数: 标签:
这是我的读书笔记,希望能够对大家学习java有所帮助。所有代码都经过测试, 测试环境 : javaversion"1.4.0-rc" Java(TM)2RuntimeEnvironment,StandardEdition(build1.4.0-rc-b91) JavaHotSpot(TM)ClientVM(build1.4.0-rc-b91,mixedmode) 如大家发现任何错误

   

这是我的读书笔记,希望能够对大家学习java有所帮助。所有代码都经过测试,测试环境

javaversion"1.4.0-rc"
Java(TM)2RuntimeEnvironment,StandardEdition(build1.4.0-rc-b91)
JavaHotSpot(TM)ClientVM(build1.4.0-rc-b91,mixedmode)

如大家发现任何错误,或有任何意见请不吝赐教。缺省构造函数的问题:base类是父类,derived类是子类,首先要说明的是由于先有父类后有子类,所以生成子类之前要首先有父类。class是由class的构造函数constructor产生的,每一个class都有构造函数,如果你在编写自己的class时没有编写任何构造函数,那么编译器为你自动产生一个缺省default构造函数。这个default构造函数实质是空的,其中不包含任何代码。但是一牵扯到继承,它的问题就出现了。如果父类baseclass只有缺省构造函数,也就是编译器自动为你产生的。而子类中也只有缺省构造函数,那么不会产生任何问题,因为当你试图产生一个子类的实例时,首先要执行子类的构造函数,但是由于子类继承父类,所以子类的缺省构造函数自动调用父类的缺省构造函数。先产生父类的实例,

然后再产生子类的实例。如下:

classbase{
}
classderivedextendsbase{
publicstaticvoidmain(String[]args){
derivedd=newderived();
}
}

下面我自己显式地加上了缺省构造函数:

classbase{
base(){
System.out.println("baseconstructor");
}
}
classderivedextendsbase{
derived(){
System.out.println("derivedconstructor");
}
publicstaticvoidmain(String[]args){
derivedd=newderived();
}
}

执行结果如下:说明了先产生baseclass然后是derivedclass。

baseconstructor
derivedconstructor

我要说明的问题出在如果baseclass有多个constructor而derivedclass也有多个constructor,这时子类中的构造函数缺省调用那个父类的构造函数呢?答案是调用父类的缺省构造函数。但是不是编译器自动为你生成的那个缺省构造函数而是你自己显式地写出来的缺省构造函数。

classbase{
base(){
System.out.println("baseconstructor");
}
base(inti){
System.out.println("baseconstructorinti");
}
}
classderivedextendsbase{
derived(){
System.out.println("derivedconstructor");
}
derived(inti){
System.out.println("derivedconstructorinti");
}
publicstaticvoidmain(String[]args){
derivedd=newderived();
derivedt=newderived(9);
}
}
D:\java\thinking\think6>javaderived
baseconstructor
derivedconstructor
baseconstructor
derivedconstructorinti

如果将base类的构造函数注释掉,则出错。

classbase{
//base(){
//System.out.println("baseconstructor");
//}
base(inti){
System.out.println("baseconstructorinti");
}
}
classderivedextendsbase{
derived(){
System.out.println("derivedconstructor");
}
derived(inti){
System.out.println("derivedconstructorinti");
}
publicstaticvoidmain(String[]args){
derivedd=newderived();
derivedt=newderived(9);
}
}
D:\java\thinking\think6>javacderived.java
derived.java:10:cannotresolvesymbol
symbol:constructorbase()
location:classbase
derived(){
derived.java:13:cannotresolvesymbol
symbol:constructorbase()
location:classbase
derived(inti){
2errors

说明子类中的构造函数找不到显式写出的父类中的缺省构造函数,所以出错。那么如果你不想子类的构造函数调用你显式写出的父类中的缺省构造函数怎么办呢?

如下例:

classbase{
//base(){
//System.out.println("baseconstructor");
//}
base(inti){
System.out.println("baseconstructorinti");
}
}
classderivedextendsbase{
derived(){
super(8);
System.out.println("derivedconstructor");
}
derived(inti){
super(i);
System.out.println("derivedconstructorinti");
}
publicstaticvoidmain(String[]args){
derivedd=newderived();
derivedt=newderived(9);
}
}
D:\java\thinking\think6>javaderived
baseconstructorinti
derivedconstructor
baseconstructorinti
derivedconstructorinti

super(i)表示父类的构造函数base(i)请大家注意一个是super(i)一个是super(8)。大家想想是为什么??

结论:子类如果有多个构造函数的时候,父类要么没有构造函数,让编译器自动产生,那么在执行子类构造函数之前先执行编译器自动产生的父类的缺省构造函数;要么至少要有一个显式的缺省构造函数可以让子类的构造函数调用。

原文转自:http://www.ltesting.net