子类对象内存结构
- super“可以看做”是直接父类对象的引用。可通过super来访问父类中被子类覆盖的方法或属性。
- 使用super调用普通方法,语句没有位置限制,可以在子类中随便调用。
- 在一个类中,若是构造方法的第一行没有调用super(……)或者this(……);那么Java默认都会调用super(),含义是父类的无参数构造方法
案例:
package com.itbaizhan.tast;
//测试super
public class TestSuper01 {
public static void main(String[] args) {
new ChildClass().f();
}
}
class FatherClass {
public int value;
public void f(){
value = 100;
System.out.println("FatherClass.value="+value);
}
}
class ChildClass extends FatherClass{
public int value;
public int age;
public void f() {
super.f();//调用父类的普通方法
value = 200;
System.out.println("ChildClass.value="+value);
System.out.println(value);
System.out.println(super.value);//调用父类的成员变量
}
public void f2() {
System.out.println(age);
}
}
继承树的追溯
属性/方法查找顺序:(比如:查找变量h)
- 查找当前类中有没有属性h
- 依此上溯每个父类,查找每个父类中是否有h,直到Object
- 如果没有找到,则出现编译错误
- 上面步骤,只要找到h变量,则这个过程终止
构造方法调用顺序: 构造方法第一句总是:super(……)来调用父类对应的构造方法,所以,流程就是:先向上追溯到Object,然后再依此向下执行类的初始化块和构造方法,直到当前子类为止
注: 静态初始化块调用顺序,与构造方法调用顺序一样,不再重复。
示例:
package com.itbaizhan.tast;
//测试树的继承
public class TestSuper2 {
public static void main(String[] args) {
new ChildClass2();
}
}
class FatherClass2{
static {
System.out.println("静态初始化:FatherClass");
}
public FatherClass2(){
super();
System.out.println("创建FatherClass2");
}
}
class ChildClass2 extends FatherClass2{
static {
System.out.println("静态初始化:ChildClass2");
}
public ChildClass2(){
super();//在一个类中,若是构造方法的第一行没有调用super(……)或者this(……);那么Java默认都会调用super(),含义是父类的无参数构造方法
System.out.println("创建ChildClass2");
}
}
总结:
- 继承,extends,子类继承父类的属性和方法,除了私有的属性和方法,以及父类的构造方法。
- 重写:override,在继承的基础上,可以改变子类,但必须在父类的范围内,或者等于父类,重写输出结果。
- Final关键字:修饰变量、方法、类,变量被修饰,不可重新赋值,方法被修饰,不可被子类重写,类被修饰,不可继承。
- 组合,在子类中新建一个类,使得父类的对象作为子类的属性,方法
- 方法的重写:equals,比较对象内容是否相等,即比较对象的地址(hashcode,)此处可以根据自己的要求重写equals方法。
- 子类对象内存结构:super,通过super来访问被子类覆盖的属性或方法,调用普通方法,没有为止限制,可以在子类中随便使用
- 继承树的追溯,当构造方法方法第一句是super(……)时,这时会继承树的追溯