【深入理解JAVA虚拟机】第二部分.内存自动管理机制.2.HotSpot虚拟机对象探秘

2、HotSpot虚拟机对象探秘

对象的创建过程

1、加载类

虚拟机遇到一条new指令时,首先将去检查这个指令的参数是否能在常量池中定位到一个类的符号引用,并且检查这个符号引用代表的类是否已被加载、 解析和初始化过。

如果没有,那必须先执行相应的类加载过程。

2、分配内存

在类加载检查通过后,接下来虚拟机将为新生对象分配内存。 对象所需内存的大小在类加载完成后便可完全确定为对象分配空间的任务等同于把一块确定大小的内存从Java堆中划分出来。

分配方式:

1、指针碰撞。适用于连续内存,需要垃圾回收有整理功能。

2、空闲列表。不需要连续内存,可用于标记-清理的垃圾回收功能。

指针操作同步问题的解决方式,则也是一般的同步问题的解决方式:

1、CAS+失败重试来保证原子性

2、线程隔离,即为每个线程分配一块单独的区域,称为本地线程分配缓冲(Thread Local Allocation Buffer,TLAB)。

虚拟机是否使用TLAB,可以通过-XX:+/-UseTLAB参数来设定。

3、内存归零

不包括对象头

4、对象头设置

5、构造函数

虚拟机部分已经完成,开始对象自定义部分。

对象的内存布局

在HotSpot虚拟机中,对象在内存中存储的布局可以分为3块区域:对象头(Header)、实例数据(Instance Data)和对齐填充(Padding)。

1、对象头

HotSpot虚拟机的对象头包括两部分信息,第一部分用于存储对象自身的运行时数据,如哈希码(HashCode)、 GC分代年龄、 锁状态标志、 线程持有的锁、 偏向线程ID、 偏向时间戳等,这部分数据的长度在32位和64位的虚拟机(未开启压缩指针)中分别为32bit和64bit,官方称它为“Mark Word”。 
对象头的另外一部分是类型指针,即对象指向它的类元数据的指针,虚拟机通过这个指针来确定这个对象是哪个类的实例。 
如果对象是一个Java数组,那在对象头中还必须有一块用于记录数组长度的数据,因为虚拟机可以通过普通Java对象的元数据信息确定Java对象的大小,但是从数组的元数据中却无法确定数组的大小。

2、实例数据

存储顺序会受到虚拟机分配策略参数(FieldsAllocationStyle)和字段在Java源码中定义顺序的影响。 HotSpot虚拟机默认的分配策略为longs/doubles、 ints、 shorts/chars、bytes/booleans、 oops(Ordinary Object Pointers),从分配策略中可以看出,相同宽度的字段
总是被分配到一起。 

3、对齐填充

对象的定位访问

Java程序需要通过栈上的reference数据来操作堆上的具体对象。 由于reference类型在Java虚拟机规范中只规定了一个指向对象的引用,并没有定义这个引用应该通过何种方式去定位、 访问堆中的对象的具体位置,所以对象访问方式也是取决于虚拟机实现而定的。

目前主流的访问方式有使用句柄和直接指针两种。 

1、句柄

使用句柄来访问的最大好处就是reference中存储的是稳定的句柄地址,在对象被移动(垃圾收集时移动对象是非常普遍的行为)时只会改变句柄中的实例数据指针,而reference本身不需要修改 
2、指针

使用直接指针访问方式的最大好处就是速度更快,它节省了一次指针定位的时间开销,由于对象的访问在Java中非常频繁,因此这类开销积少成多后也是一项非常可观的执行成本。


HotSpot是使用第指针方式进行对象访问的

内存溢出

 Java堆溢出

java.lang.OutOfMemoryError:Java heap space

通过查看引用链确定溢出原因:

内存合理的大:内存溢出   调大内存,或调小业务内存模型

内存不合理的大:内存泄漏  解决问题

虚拟机栈或本地方法栈溢出

关于虚拟机栈和本地方法栈,在Java虚拟机规范中描述了两种异常:
如果线程请求的栈深度大于虚拟机所允许的最大深度,将抛出StackOverflowError异常。
如果虚拟机在扩展栈时无法申请到足够的内存空间,则抛出OutOfMemoryError异常。 
线程过多,会导致内存溢出。

方法区和运行时常量池溢出

Exception in thread"main"java.lang.OutOfMemoryError:PermGen space 
方法区溢出也是一种常见的内存溢出异常,一个类要被垃圾收集器回收掉,判定条件是比较苛刻的。

本机直接内存溢出

DirectMemory容量可通过-XX:MaxDirectMemorySize指定,如果不指定,则默认与Java堆最大值(-Xmx指定)一样 
由DirectMemory导致的内存溢出,一个明显的特征是在Heap Dump文件中不会看见明显的异常,如果读者发现OOM之后Dump文件很小,而程序中又直接或间接使用了NIO,那就可以考虑检查一下是不是这方面的原因

原文地址:https://www.cnblogs.com/aoyihuashao/p/10344737.html

时间: 2025-01-16 08:22:16

【深入理解JAVA虚拟机】第二部分.内存自动管理机制.2.HotSpot虚拟机对象探秘的相关文章

【深入理解JAVA虚拟机】第二部分.内存自动管理机制.3.垃圾收集器与内存分配策略

1.学习目的 当需要排查各种内存溢出. 内存泄漏问题时,当垃圾收集成为系统达到更高并发量的瓶颈时,我们就需要对这些“自动化”的技术实施必要的监控和调节. Java内存运行时区域的各个部分,其中程序计数器. 虚拟机栈. 本地方法栈3个区域随线程而生,随线程而灭:栈中的栈帧随着方法的进入和退出而有条不紊地执行着出栈和入栈操作. 因此这几个区域的内存分配和回收都具备确定性,在这几个区域内就不需要过多考虑回收的问题,因为方法结束或者线程结束时,内存自然就跟随着回收了. 而Java堆和方法区则不一样,一个

内存自动管理

内存自动管理    20 内存模型    20 Jvm垃圾收集算法    22 标记-清除    22 标记-整理    22 复制算法    22 分代思想    23 Stop-The-World    23 java堆溢出:    24 方法区和运行时常量池溢出    27 本机直接内存溢出    29 垃圾收集器    30 引用计数器:    30 可达性分析算法    31 回收方法区    32 垃圾收集器    33 HotSpot垃圾回收器    33 l CMS Failur

Cocos2d-x内存自动释放机制--透彻篇

首先在架构里面需要明白,如果使用new创建对象的话,我们需要自己释放内存,如果直接用引擎提供的警静态方法,我们可以不做内存管理,引擎自动处理,因为引擎背后有一个自动释放池.通过查看源码可以知道,每个静态方法都会调用autorelease()方法.如果我们需要引擎自动释放一个对象,创建后可以调用该对象的autorelease()(前提是该对象所属类继承自CCObject),该方法会把当前对象放入自动释放池中,每次帧过渡时都会去遍历检查是否可以释放掉该对象的内存. 源码如下: // 创建一个精灵对象

Oracle 11g设置内存自动管理

Oracle 11g设置内存自动管理: ALTER SYSTEM SET MEMORY_TARGET = 1024M SCOPE=SPFILE;ALTER SYSTEM SET memory_max_target = 1500M SCOPE=SPFILE;ALTER SYSTEM SET SGA_TARGET = 0 SCOPE=SPFILE;ALTER SYSTEM SET SGA_MAX_SIZE=800M SCOPE=SPFILE;ALTER SYSTEM SET PGA_AGGREGAT

Senparc.Weixin.MP SDK 微信公众平台开发教程(十六):AccessToken自动管理机制

Senparc.Weixin.MP SDK 微信公众平台开发教程(十六):AccessToken自动管理机制 在<Senparc.Weixin.MP SDK 微信公众平台开发教程(八):通用接口说明>中,我介绍了获取AccessToken(通用接口)的方法. 在实际的开发过程中,所有的高级接口都需要提供AccessToken,因此我们每次在调用高级接口之前,都需要执行一次获取AccessToken的方法,例如: 1 var accessToken = AccessTokenContainer.

Android内存进程管理机制

参考文章: http://www.apkbus.com/android-104940-1-1.htmlhttp://blog.sina.com.cn/s/blog_3e3fcadd0100yjo2.html 一.理论: Android采取了一种有别于Linux的进程管理策略,有别于Linux的在进程活动停止后就结束该进程,Android把这些进程都保留在内存中,直到系统需要更多内存为止.这些保留在内存中的进程通常情况下不会影响整体系统的运行速度,并且当用户再次激活这些进程时,提升了进程的启动速度

【深入理解Java原理】 JVM 内存区域 &#357506;

原文: http://blog.gqylpy.com/gqy/444 置顶:来自一名75后老程序员的武林秘籍--必读(博主推荐) 来,先呈上武林秘籍链接:http://blog.gqylpy.com/gqy/401/ 你好,我是一名极客!一个 75 后的老工程师! 我将花两分钟,表述清楚我让你读这段文字的目的! 如果你看过武侠小说,你可以把这个经历理解为,你失足落入一个山洞遇到了一位垂暮的老者!而这位老者打算传你一套武功秘籍! 没错,我就是这个老者! 干研发 20 多年了!我也年轻过,奋斗过!我

深入理解Linux内核 - 第二章 内存寻址 04

问题:段描述符中的字段含义? 段首地址的线性地址/偏移/粒度/特权限制等信息. 1,快速访问段描述符 在加载段选择符时,相应的段描述符被自动由内存装入对应的非编程CPU寄存器.装入后就不再访问主存中的GDT或LDT. 2,段选择符的字段 index:指定了放在GDT或LDT中相应段描述符的入口 TI:0 GDT 1 LDT RPL:特权级,特指装入CS段中当前CPU的特权级. 3,段描述符地址换算 GDT地址:0x00020000 index:2 地址:0x00020000+(2*8))0x00

深入理解Linux内核 - 第二章 内存寻址 03

问题:1,逻辑地址怎么求?2,段描述符的base字段怎么求? 逻辑地址:由16位段选择符和32位偏移量组成,并不是说这个地址就是段选择符偏移32位再和偏移量相加. 段选择符:或叫段标识符,是一个16位长的字段.顾名思义,用于选择段的.为了快速找到段选择符,处理器提供段寄存器. 15~3bit,段索引号,2bit TI位, 1 0 位,特权级别. 由于段选择符仅仅在转换成线性地址中发挥作用,则通常认为32位偏移量就是逻辑地址.也即c语言中用&求得一个变量的地址就是逻辑地址. 段描述符:段选择符中,