一、继承中的构造方法的执行顺序
- 创建对象时,会先自动调用父类的构造函数,再自动调用子类的构造函数。
- 解释:super()表示父类的构造函数并会调用于参数相对应的父类中的构造函数。子类中,它在调用父类中空参数的构造函数。因为子类继承父类,会继承到父类中的数据,所以必须要看父类是如何对自己的数据进行初始化的。因此子类在进行对象初始化时,先调用父类的构造函数,这就是子类的实例化过程。
- 总的来说,初始化顺序依次是:(静态变量、静态初始化块)–>(变量、初始化块)–> 构造器;如果有父类,则顺序是:父类static方法 –> 子类static方法 –> 父类构造方法- -> 子类构造方法
二、子类的所有构造函数中的第一行,其实都有一条 隐身的语句super();
1、父类有空参:没有有参构造时系统默认给出了,或者有有参构造并且也给出了空参构造
- 则此时子类自动调用父类 构造函数即可。
2、若父类没有空参:有有参构造但未给出空参构造
- 注意:构造方法不是静态的,即此时编译看左边,运行看右边,也就是若父类没有空参构造,而子类在构造函数的第一句又会默认调用父类的空参构造即隐身的super(),则此时编译不能通过。
- 解决办法:
1)指定显示调用父类的有参构造super(a,b) ,此时就不会再隐式调用super();
2)若子类构造函数中用this来指定调用子类自己的有参构造函数this(a,b),那么被调用的构造函数也一样会访问父类中的有参构造函数。
3)注意:构造函数的第一句,一定是super或者this,只能选择一种。
补充:
假设有两个类:
Person:父类,成员变量:name,age
Student:子类,继承Person,则它的成员变量name,age,假设再新增一个成员变量address
那么当Student写构造函数时:
public Student(String name,int age,String address){
this.name = name;
this.age = age;
this.address =address;
}
当看到这段代码时,我们可以想到,name和age的赋值跟Person父类构造函数里面的name、age赋值是重复的,有重复代码,所以为了减少重复代码,java采用super关键字调用父类的构造函数来减少重复代码。(子类无法继承父类的构造函数,可以继承成员变量和方法)
向上转型:
1.一个引用能够调用哪些成员(变量和方法),取决于这个引用的类型究竟是怎么定义的。
Student s = new Student();
Person p = s;
如果Person有一个方法introduce()
Student有两个方法study()和重写(override)的introduce()
则p只能调用introduce方法不能调用study方法.
2.一个引用调用的是哪一个方法,取决于这个引用所指向的对象。
同上例子,则p调用introduce方法,会调用Student的introduce。
3.总结: Person p = new Student();
p能够调用哪些成员看左边的类型——p实际调用的哪个方法看右边的实际对象。
向下转型
前提:必须先把一个对象向上转型才能进行向下转型给转回来,如:
Person p = new Student();
Student s = (Student)p;
上述转型是正确的
而:
Person p = new Person();
Student s = (Student)p;
这种做法是错误的。(理解:比如一个学生,你说他是人这是没问题的,但是如果一个人你说他是学生那就不一定了,所以上述那种转型做法是不正确的)