众所周知,函数调用在内存中是通过压栈,退栈实现的,而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/

转载于:https://www.cnblogs.com/whongfei/p/5246956.html

Java对象的内存模型相关推荐

  1. java对象内存模型_Java对象的内存模型

    众所周知,函数调用在内存中是通过压栈,退栈实现的,而Java的方法调用则是在JVM栈中通过栈帧实现的,且所有的Java对象都只在堆上分配内存.那么一个Java对象在堆内存里到底长啥样呢?实际上,当一个 ...

  2. JAVA基础驿站,Java基础:内存模型

    在java基础当中,有很重要一块的内容,就是关于java当中的内存模型,所涉及的问题无非就是关于java当中的类.实例.变量.常量.静态变量.方法等blabla的东西都分别存储在什么地方.这部分的内容 ...

  3. java关键字 valotile_Java内存模型-jsr133规范介绍,java中volatile关键字的含义

    最近在看<深入理解Java虚拟机:JVM高级特性与最佳实践>讲到了线程相关的细节知识,里面讲述了关于java内存模型,也就是jsr 133定义的规范. 系统的看了jsr 133规范的前面几 ...

  4. 原创 | 灵魂拷问:Java对象的内存分配过程是如何保证线程安全的?

    △Hollis, 一个对Coding有着独特追求的人△ 这是Hollis的第 222 篇原创分享 作者 l Hollis 来源 l Hollis(ID:hollischuang) JVM内存结构,是很 ...

  5. 学习笔记【Java 虚拟机④】内存模型

    若文章内容或图片失效,请留言反馈.部分素材来自网络,若不小心影响到您的利益,请联系博主删除. 总目录 学习笔记[Java 虚拟机①]内存结构 学习笔记[Java 虚拟机②]垃圾回收 学习笔记[Java ...

  6. java happens before_java内存模型以及happens-before规则

    原创文章&经验总结&从校招到A厂一路阳光一路沧桑 image# 1. JMM的介绍 # 在上一篇文章中总结了线程的状态转换和一些基本操作,对多线程已经有一点基本的认识了,如果多线程编程 ...

  7. Java 对象占用内存大小

    Java 对象 如果想要了解java对象在内存中的大小,必须先要了解java对象的结构. HotSpot虚拟机中,对象在内存中存储的布局可以分为三块区域:对象头(Header).实例数据(Instan ...

  8. 独占设备的分配与回收_灵魂拷问:Java对象的内存分配过程是如何保证线程安全的?...

    点击上方"linkoffer", 选择关注公众号高薪职位第一时间送达 作者 l Hollis JVM内存结构,是很重要的知识,相信每一个静心准备过面试的程序员都可以清楚的把堆.栈. ...

  9. jvm学习笔记(2)——java对象的内存布局

    一.内存布局: 一个java对象在内存中可分为3个区域:对象头(Header).实例数据(Instance Data).对齐填充(Padding). 对象头(两部分): 对象自身运行时的数据.如哈希码 ...

最新文章

  1. eclipse常用插件介绍
  2. 瞬间几千次的重复提交,我用 SpringBoot+Redis 扛住了!
  3. Medoo入门:安装和配置-Medoo使用指南
  4. python.freelycode.com-最难搞的python“反面”代码
  5. 思科携手中兴掌握3G话语权 剑指华为3COM组合
  6. 科学家研究:生女有撇步 多钙少碰香蕉
  7. 镶套iframe 鼠标滚动无效解决办法
  8. POS-商户手续费-从生活剖析,通俗易懂
  9. java使用btree_最新阿里、百度Java面试题(基础+框架+数据库+分布式+JVM+多线程)...
  10. intellij idea 主题下载网站
  11. MFC 鼠标光标资源 及 更换
  12. 07 ,矩阵的转置,矩阵的行列式,方阵 ( 2阶行列式,3阶行列式,n 阶行列式 ) :
  13. About all the awsome websites
  14. Day06(上)C++继承和派生
  15. 不用CorelDraw怎么编辑CDR文件
  16. Windows Shell编程-第十章.Winows辅助库
  17. uniapp去掉返回键
  18. linux module load, show, 包管理,找到包的路径
  19. oppo手机便签存储路径在哪
  20. 计算表面静电势_APBS工具

热门文章

  1. 日、德、美、中各国“工业4.0“们的核心差异
  2. 我为什么暂时放弃了React Native
  3. windows如何卸载Oracle
  4. 超棒的跨浏览器纯CSS动画实现 - Animate.css
  5. [转贴]NIOS II 软核中EPCS配置芯片的存储操作
  6. 学习Exchange管理最佳实践
  7. 京东Vue组件库NutUI 2.0发布:将支持跨平台!
  8. stm32 堆和栈(stm32 Heap Stack)
  9. __sync_fetch_and_add
  10. linux内核配置系统浅析(转)