首先,我们定义两个类A,B,其中B extends A
class A {
static {
int x=3;
}
String num = "父类成员变量";
public void show() {
System.out.println(this.num);
System.out.println(this.function1());
}
public String function1() {
System.out.println(this.num);
return "父类调用";
}
}
class B extends A {
static {
String str="123";
}
String num = "子类成员变量";
public String function1() {
System.out.println(this.num);
return "子类调用";
}
}
然后我们再写个测试类Demo
public class Demo {
public static void main(String[] args) {
A a = new B();
System.out.println(a.num);
System.out.println(a.function1());
a.show();
}
}
这里使用向上转型A a=new B(),表示父类的引用指向子类的对象。我们在main方法中打上断点,开始查看A a=new B()的执行流程。
一开始,程序会最先进入到父类的静态块,因为静态的成员或者方法不属于对象,是属于类的,同时它只加载一次。所以,即A类的static中,int x=3 会最先加载到jvm的方法区中。然后再加载子类的静态块,即B类的String str="123"会接着加载到方法区中。
之后,代码会进入到父类的成员变量,即A类的String num="父类成员变量",这个num是字符串,保存在字符串常量池中了,之后代码继续运行,会把子类B类的String num="子类成员变量"加载到字符串常量池中。此时,A a=new B() 执行完毕。我们可以查看对象a,发现里面存在两个num,分别就是之前加载过的父类成员变量和子类成员变量。
这里,说明一点,我们通过a+"."可以有代码提示,此时提示出来的方法和变量是A类的本身的方法和变量,以及Object类的信息,但是我们要明白,如果是a.function1(),由于子类重写父类的同名方法,此时执行的是B类的function1(),但是如果是a.show(),子类没有重写show()方法,就会执行父类本身的show().关于a.num,由于成员变量不存在多态,所以变量num是对象a所属类A中的成员变量num。