本篇文章总结了一些关于Java构造方法的常见问题。
为什么创建了一个子类对象会调用其父类的构造方法
如下代码示例:
package simplejava; class Super { String s; public Super() { System.out.println("Super"); } } class Sub extends Super { public Sub() { System.out.println("Sub"); } } public class Q4 { public static void main(String[] args) { Sub s = new Sub(); } }
结果打印:
Super
Sub
当一个类继承了某个类时,在子类的构造方法里,super()必须先被调用;如果你没有写,编译器会自动调用super()方法,即调用了父类的构造方法;
这并不是创建了两个对象,其实只有一个子类Sub对象;之所以需要调用父类的构造方法是因为在父类中,可能存在私有属性需要在其构造方法内初始化;
当编译器自动插入父类构造方法的调用后,子类的构造方法类似如下代码:
public Sub() { super(); System.out.println("Sub"); }
一个常见的错误信息:Implicit super constructor is undefined for default constructor
对于Java开发人员来说,这是一个很常见的编译错误:
"Implicit super constructor is undefined for default constructor. Must define an explicit constructor"
之所以出现这个编译错误,是因为父类的默认构造方法未定义。在Java中,如果一个类没有定义构造方法,编译器会默认插入一个无参数的构造方法;但是如果一个构造方法在父类中已定义,在这种情况,编译器是不会自动插入一个默认的无参构造方法,这正是以上demo的情况;
对于子类来说,不管是无参构造方法还是有参构造方法,都会默认调用父类的无参构造方法;当编译器尝试在子类中往这两个构造方法插入super()方法时,因为父类没有一个默认的无参构造方法,所以编译器报错;
要修复这个错误,很简单:
1、在父类手动定义一个无参构造方法:
public Super(){ System.out.println("Super"); }
2、移除父类中自定义的构造方法
3、在子类中自己写上父类构造方法的调用;如super(value);
在子类中明确调用父类构造方法
以下代码运行正常:
父类定义了一个有参构造方法,然后子类构造方法明确调用了父类有参构造方法;
规则总结
简单的说,规则是这样的:
在子类构造方法中,不管是程序员手动调用父类构造方法还是编译器自动调用,
子类构造方法必须要调用父类的某个构造方法;
被子类调用的父类构造方法在父类中必须是存在的;
一个有趣的问题
如果一个类定义了一个有参数的构造方法,Java为什么就不提供默认的无参构造方法呢?
有兴趣的可以看下stackoverflow中的回答:
http://stackoverflow.com/q/16046200/127859