java中父类子类的成员变量和成员方法在内存分配上的一些理解

先声明,以下是个人自己的理解(我也怕误人子弟),如有错误请指出一起学习交流。

先上代码:

package myTest;

class A {
	int num = 5;

	A() {
		show();
	}

	public void show() {
		System.out.println("A" + num);
	}
}

class B extends A {
	int num = 10;

	B() {
		show();
	}

	public void show() {
		 System.out.println("B"+num);
		//super.show();
	}
}

public class Test01 {
	public static void main(String[] args) {
		B b = new B();
	}
}

以上代码输出:

B0

B10

个人理解:在new B()的时候,肯定先在B的堆区中构造父类的实例,按照顺序是 :父类的成员变量num,并赋值等于5-> 父类的无参构造函数(若指定构造函数另说) -> 子类的成员变量并赋值 -> 子类的构造函数,这个顺序,所以B的堆区中有两部分,一部分是this区域,代表B继承过来的变量和方法,还有一部分是super区域,代表父类的变量和方法。

所以,在new B()的时候,先执行父类的构造函数,由于程序会给每一个方法隐藏传入this变量,所以这里的this变量肯定是B,由于B重写的show()方法,所以调用B的show()方法,因为在执行父类构造函数的时候,B的num还没有初始化,所以是0。然后初始化B的num,再执行B的构造方法,此时也是调用B的show(),并且此时num的值已经是10。

注意:如果B类不重写A类的show(),那么B的堆区this区域中也没有执行show()代码区的地址,所以此时this找不到show()会自动去找super区域的show(),这种情况会输出A5。变量没有重写之说,继承过来的变量都会在B的this区域分配内存,就算B中没有为num赋值,在B的this区域还会为num分内存。若A有私有变量,会在B的堆区的super区域分配内存,只是对外不公开,所以被sun公司定义为私有变量不能被继承(个人猜测)。

大家可以尝试一下B中不重写show()和不给num赋值,看看效果就理解了。


请看在内存中:


猜你喜欢

转载自blog.csdn.net/fxxashelly/article/details/80872466