为什么在创建子类对象的时候会调用父类的构造函数
原文:Constructors of Sub and Super Class in Java?
1.为什么在创建子类对象的时候会调用父类的构造函数?
public class Sub extends Super{
public Sub() {
System.out.println("Sub");
}
public static void main(String[] args) {
Sub s = new Sub();
}
}
class Super {
String s;
public Super() {
System.out.println("Super");
}
}
这段代码的输出结果为:
当一个类继承了其它类时,在它的构造函数(constructor)中super()
必须被首先调用,如果super()
没有被调用,则编译器将在构造函数(constructor)的第一行插入对super()
的调用。这就是为什么当创建一个子类的对象时会调用父类的构造函数(constructor)的原因。
这不是创建两个对象,仅创建了一个子对象。父类的构造函数被调用是考虑到其可能有私有的属性需要通过自身的构造函数初始化。
编译器在子类的构造函数中插入对父类构造函数的引用后,子类的构造函数看起来像下面的代码
public Sub() {
super();
System.out.println("Sub");
}
2.一个常见的错误信息:父类的默认构造函数未被定义
这个错误产生的原因是父类的默认构造函数为被定义。在java中,如果一个类没有定义过一个构造函数,这编译器将为这个类添加一个默认的无参构造函数。如果这个类已经定义了一个构造,像这种Super(String s)
,编译器将不会添加默认的无参构造函数。
子类的构造函数(有参或无参),都将调用父类的无参构造函数。由于编译器尝试为子类的两个构造函数添加super()
调用,但父类的默认构造函数没有被定义,编译器报错。
解决这个问题,有几种方式:
a.像下面这样为父类添加无参的构造函数
public Super() {
System.out.println("Super");
}
b.移除父类中自定义的构造函数
c.在子类的构造函数中添加super(value)
调用
3.在子类的构造函数中显示调用父类的构造函数
public class Sub extends Super{
int x = 200;
public Sub(String s) {
super(s);
}
public static void main(String[] args) {
Sub s = new Sub("a");
}
}
class Super {
String s;
public Super(String s) {
this.s = s;
System.out.println("Super s");
}
}
4.规则
简短地说:子类的构造函数必须引用父类的构造函数,由程序猿显示调用或由编译器隐式调用,对于这两种方式,被引用的父类构造函数必须已被定义。
上一篇: php中Redis的应用--消息传递
下一篇: 浅谈FileItem类的常用方法