先上问题:
为什么我的子类无参调用父类有参会出错??
有同学这样问我,上图。
class Person{
public Person(String name)
{
System.out.println("My name is "+name);
}
}
class Man extends Person{
private String name = "Hello";
public Man(){
super(name); //这里调用父类有参出错
}
}
public class TestDemo {
public static void main(String []args) {
Man m = new Man();
}
}
可以看到 子类的无参构造调用父类有参构造是出错了。编译器不允许这样做。
其实这是一个历史问题,历史老师是这样说的:
0.构造方法一般是为对象的某些属性作初始化的。
1. 我们先看看一般怎样创建子类。Man m = new Man();
当new一个Man对象时,是会调用该类的无参构造方法的。
2. java继承中,执行子类的构造方法之前,会先调用父类中没有参数的构造方法。如果没有显式的super调用,那么就是默认调用父类无参。 其目的是为了帮助继承自父类的成员作初始化操作。
目前以上三点能告诉我们以上代码非static构造方法的执行顺序了。
那么这个问题其实是关于类初始化的问题。不禁想:既然无法在超类构造器之前引用name,那么是不是name还没初始化呢?那么有这想法就好办了,在name属性前加一个static试试。
确实可以了。那么实际类初始化顺序是怎样的呢?
- 类的初始化顺序:
1). 父类static(顺序按照代码书写顺序进行)
2). 子类static(顺序按照代码书写顺序进行)
3). 父类非static域、构造器(顺序按照代码书写顺序进行)
4). 子类非static域、构造器(顺序按照代码书写顺序进行)
也就是说,在父类构造器加载时(相当于参数赋值时)检查到name(因为是在子类中)其实并未有name的加载入,所以jvm检测到就报错。那么在name属性前加个static就是让name先加载(赋值)了先,然后再在父类构造器里引用那就没有问题了。
- 目前对这个问题了解到这么多