10.1.继承的细节问题:
- 子类继承了所有的属性和方法,非私有的属性和方法可以在子类直接访问,但是私有属性和方法不能在子类直接访问,要通过父类提供公共的方法去访问
- 子类必须调用父类的构造器,完成父类的初始化
- 当创建子类对象时,不管使用子类的哪个构造器,默认情况下总会去调用父类的无参构造器,如果父类没有提供无参构造器,则必须在子类的构造器中用 super 去指定使用父类的哪个构造器完成对父类的初始化工作,否则,编译不会通过。
父类的有参构造器存在情况下,若没声明无参构造器(默认构造器),则无参构造器会被覆盖
public class test{ public test(){} //在这种情况下,若不声明,则会被覆盖 public test(int a,int b){...} }
- 如果希望指定去调用父类的某个构造器,则调用方式为:
super(参数列表)
若父类有无参构造器,用
super()
调用(默认,可省略),所以,若子类构造器没有super,则默认调用 super() 来进行无参调用
- super在使用时,必须放在构造器的第一行
super(…) 只能在构造器中使用,注意,不是super
- super() 和 this 都只能放在构造器第一行,因此这两个方法不能共存在一个构造器
- Java所有类都是 Object 类的子类,Object 是所有类的基类(都继承它)
- 父类构造器的调用不限于直接父类!将一直往上追溯直到Object类(顶级父类)
一直往上追溯的继承类中构造器都会加载,并且加载顺序为由最顶级开始往下依次加载,即Object -> … -> 爷爷类 -> 父类 -> 子类
- 子类最多只能继承一个父类(直接继承),即java中是单继承机制
- 不能滥用继承,子类和父类之间要满足从属的逻辑关系
即
Cat extends Person
语句没毛病,但是逻辑不通
Cat extends Animal
逻辑比较合理
10.2.继承的内存布局
可以看出,在创建一个对象时,该子类的被继承类都要完成初始化
10.3.访问对象属性和方法的继承机制
在调用对象的属性或者方法时,继承机制为:
- 看子类是否有该属性或者方法
- 如果子类有这个属性或者方法,并且可以访问(访问修饰符限制),则返回信息
若子类有这个属性或者方法,但是不可以访问(private),系统也不会继续往父类找,因为已经找到了,只是不可以访问,所以会提示该属性或者方法不可访问
- 如果子类没有这个属性,就看父类有没有这个属性或者方法,如果父类有该属性或者方法,并且可以访问,就返回信息
- 如果父类没有就按照(3)的规则,继续找上级父类,直到 Object(万类之初)
10.4.super基本介绍
super 代表父类的引用(直接跳过子类),用于访问父类的属性、方法、构造器
10.5.super基本语法
- 访问父类的属性,但不能访问父类的private属性:
super.属性名
- 访问父类的方法,但不能访问父类的private方法:
super.方法名(参数列表)
- 访问父类的构造器:
super(参数列表)
只能放在构造器的第一句,只能出现一句
10.6.super 真香定理
- 分工明确,父类属性由父类初始化,子类属性由子类初始化
- 当子类中有和父类中的成员(属性和方法)重名时,为了访问父类的成员,必须通过super。如果没有重名,使用super、this、直接访问是一样的效果(指访问父类的成员)
- super 的访问不限于直接父类,如果爷爷类和本类中有同名的成员,也可以使用 super 去访问爷爷类的成员;如果多个基类(上级类)中都有同名的成员,使用super访问遵循就近原则,也需要遵循访问权限的相关规则。
10.7.super和this的比较
区别点 | this | super |
---|---|---|
访问属性 | 访问本类中的属性,如果本类没有此属性则从父类中继续查找 | 从父类开始查找属性 |
调用方法 | 访问本类中的方法,如果本类没有此方法则从父类中继续查找 | 从父类开始查找方法 |
调用构造器 | 调用本类构造器,必须放在构造器的首行 | 调用父类构造器,必须放在子类构造器的首行 |
特殊 | 表示当前对象 | 子类中访问父类的对象 |