一、继承中的构造方法的执行顺序

  • 创建对象时,会先自动调用父类的构造函数,再自动调用子类的构造函数。
  • 解释: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; 

这种做法是错误的。(理解:比如一个学生,你说他是人这是没问题的,但是如果一个人你说他是学生那就不一定了,所以上述那种转型做法是不正确的)