Java对象的内存模型

众所周知,函数调用在内存中是通过压栈,退栈实现的,而Java的方法调用则是在JVM栈中通过栈帧实现的,且所有的Java对象都只在堆上分配内存.那么一个Java对象在堆内存里到底长啥样呢?实际上,当一个对象在内存中被创建的时候,它只不过是一串0和1而已.编译器会维护一张表,这张表用来存储对象中的每一个成员变量所在位置的偏移量(offset).这样,通过查这张表,JVM就能知道每一个成员变量相对于其起始地址所在的位置了.


来看这样一个例子.我们定义一个名为 Base 的类,它没有任何的成员方法,只有两个成员变量x和y.Base 对象的内存模型如下图所示:




然后我们从Base类派生一个名为 Derived 的子类,则 Derived 对象的内存模型如下:




可以看到,子类的内存模型实际上就是在父类的基础上添加了子类特有的成员变量而已.这样设计的好处是,如果有一个 Base 类型的引用指向了 Derived 对象,那么由于 Derived对象的内存模型中包含了其父类 Base,因而 Base 类对于其子类 Derived 是可见的. 这样一来,任何通过Base引用操作 Derived 对象的调用都是安全的.

啥叫安全呢?例如上面这个例子,编译器会维护一张保存有成员变量offset的表,这张表是这么写的:x变量在第1个位置,y变量在第2个位置,z变量在第3个位置.(当然实际中肯定不是第几个位置这么简单,编译器会根据变量的数据类型确定其offset,如,一个int偏移4字节,一个double偏移8字节).当我们通过Base引用来引用Derived中的成员时,编译器就会去找这个对象中的第几个位置.比如调用derived.x 会去找第一个位置,derived.z,则会去找第三个位置.因为子类Derived中包含了父类的x,y变量,而且次序正好排在x,y之后,所以derived.z可以被成功执行.


按照这个逻辑,方法也可以放在每个对象的起始位置:



不过,这么干是非常低效的.如果一个类有很多的方法,那么就要在起始位置保存大量的数据,并且每个对象都会重复地保存这些函数代码.这样对象构造起来就慢,造成空间和时间上的性能浪费.


解决这个问题的一个办法是,为每一个类创建一个虚表(virtual table),这张表里保存了这个类中所有的方法代码.而对于这个类的对象,则在其内存的起始位置中保存一个指向此表的指针.这样一来,多个对象就能共享一份方法代码了.



英文原文:http://www.programcreek.com/2011/11/what-do-java-objects-look-like-in-memory/

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java对象内存模型包括对象头、实例数据和对齐填充三部分。对象头包含了关于堆对象的布局、类型、GC状态、同步状态和标识哈希码等基本信息。实例数据存放了类的数据信息、父类的信息以及对象字段属性信息。对齐填充主要是为了字节对齐而填充的数据,以凑齐8字节的倍数。 在JVM中,我们可以使用openjdk的jol工具来打印对象信息。通过调用`ClassLayout.parseInstance(object).toPrintable()`方法,我们可以打印出对象的信息。 例如,对于一个无属性的对象,打印的信息如下: ``` java.lang.Object object internals: OFFSET SIZE TYPE DESCRIPTION VALUE 0 4 (object header) 01 00 00 00 (00000001 00000000 00000000 00000000) (1) 4 4 (object header) 00 00 00 00 (00000000 00000000 00000000 00000000) (0) 8 4 (object header) e5 01 00 f8 (11100101 00000001 00000000 11111000) (-134217243) 12 4 (loss due to the next object alignment) Instance size: 16 bytes Space losses: 0 bytes internal + 4 bytes external = 4 bytes total ``` 这段信息描述了对象的各个部分。例如,对象头占据了前12个字节,分为三部分,每部分占4个字节。实例数据为空,所以没有具体的值。 这就是Java对象内存模型。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [Java对象内存模型](https://blog.csdn.net/u013190417/article/details/122532408)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值