假设有两个类:
Person:父类,成员变量:name,age
Student:子类,继承Person,则它的成员变量name,age,假设再新增一个成员变量address
那么当Student写构造函数时:
- 1
- 2
- 3
- 4
- 5
当看到这段代码时,我们可以想到,name和age的赋值跟Person父类构造函数里面的name、age赋值是重复的,有重复代码,所以为了减少重复代码,java采用super关键字调用父类的构造函数来减少重复代码。(子类无法继承父类的构造函数,可以继承成员变量和方法)
super()为什么要放在构造函数第一行呢,不放在第一行编译会报错, 试想如果不放在第一行,那么我成员变量都没初始化,就对成员变量进行操作是不是会产生异常呢,如NullPointerException等等
//*****************
向上转型:
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;
这种做法是错误的。(理解:比如一个学生,你说他是人这是没问题的,但是如果一个人你说他是学生那就不一定了,所以上述那种转型做法是不正确的)
原因:
子类的所有构造函数中的第一行,其实都有一条隐身的语句super();解释:
super()表示父类的构造函数并会调用于参数相对应的父类中的构造函数。子类中,它在调用父类中空参数的构造函数。因为子类继承父类,会继承到父类中的数据,所以必须要看父类是如何对自己的数据进行初始化的。所以子类在进行对象初始化时,先调用父类的构造函数,这就是子类的实例化过程。