首页 > java向上转型时堆上发生了什么?

java向上转型时堆上发生了什么?

javaclass Base{
    public int a ;
    public int b;
    public void f();
}

class A extends Base{
    public int c;
}

如上面的两个类
定义
A a = new A();
Base b = (Base) a;
Base c = a;
当定义a时,内存中在栈上存储对A对象的引用a,在堆上生成A对象
当定义b时,内存中在栈上存储Base对象的引用,那么在堆上会是什么样的变化?
当定义c时,栈和堆上分别有什么生成?


定义c的时候栈上还是Base对象的引用,这个是在编译的时候确定的,但是堆上会有什么变化,个人猜测应该没啥变化吧,还真是比较偏的问题,有空研究研究


java里对象的内存分布并没有在标准里规定, 是和 具体jvm实现相关的. jvm spec 8, section 2.7

对sun java来说, 你的代码, 在堆里只有一个对象, 就是class A. 具体在内存里, 应该有 java 对象头(指定此对象对应的class), 各个fields(包含其parent class的各个域), 还有padding. 有兴趣的话, 可以去玩一玩UnSafe, 看看java对象的内存分布. 随便找一个link:
http://ifeve.com/sun-misc-unsafe/

然后你定义的三个引用, 其实都指向同一个A对象. 这些引用, 只在编译器有意义. 这么说吧, 比如Base和A 里都有f()函数, 然后你用这三个引用去调用f(), 最终调用的都是 A里的f(). 因为 上面说了, 此java对象, 在java对象头中包含了 它真正是哪个class的信息.


java在调用子类的构造方法之前会调用父类的构造方法,所以 new A()之后,堆上有A和Base两个对象,Base b =(Base)a,应该会把生成的Base对象的引用赋值给b,对于 c=a,其实c是指向 A的。

【热门文章】
【热门文章】