JVM学习总结——九、JVM中如何创建对象及内存分配过程

创建对象:当java虚拟机遇到new 指令时。
                 首先检查指令参数能否在常量池中定位到一个类的引用、加载、解析、初始化。
                  如果没有那必须先执行相应的类加载过程。

指针碰撞:使用过的内存被分到一边,空闲内存放到另一边。中间放着一个指针作为分解点指示器。指针移动来分配。
                   仅通过指针移动位置分配对象,并发情况会出现线程不安全,两种解决方案:
                  1.分配内存空间动作进行同步处理,实际上JVM采用CAS配上失败重试保证更新操作原子性。
                  2.按照线程划分不同的空间进行,每个线程预先分配一小块内存(本地缓冲 TLAB),
                    每个线程在自己缓冲区中创建如果使用完了,需要同步锁定重新分配(通过-XX:+/-UseTLAB 设置) 
                    对TLAB空间中无法分配的对象,JVM会尝试在Eden空间中进行分配。
                    如果Eden空间无法容纳该对象,就只能在老年代中进行分配空间。                

空闲列表:已被使用的内存和空闲内存交错在一起,虚拟机就必须维护一个列表,记录内存是否可用。通过列表找空闲内存。

对象分配完成之后需要初始化为零,初始化部分不包括对象头部分。如果使用TLAB方式,初始过程会提前至TLAB分配时进行。

对象的这两种分配方式取决于垃圾回收器算法。
serial、pernew等带压缩整理过程的收集器,系统采用指针碰撞方式分配对象空间,简单高效
CMS这种基于清除算法的收集器,系统采用空闲列表方式分配对象空间。

对象在堆中构造:对象头、实例数据、对齐填充。

对象头包含两部分:
1.Mark Word 用于存储对象自身运行数据(哈希码,GC分代年龄,锁状态标志,线程持有锁偏向线程ID,偏向时间戳)
2.类型指针,对象指向它类型元数据指针,JVM通过指针确定该对象是个类实例。
对象如果是java数组对象头还有一块记录数组长度数据。

实例数据:对象正在存储的有效信息,代码定义的各种类型字段不管继承来的还是子类本身的。
                  这些字段存储顺序 (-XX:FieldsAllocationStyle设置)也和源码定义顺序有关。默认顺序longs/doubles,ints,shorts

对齐填充:仅仅起着占位符作用。

对象访问方式

java程序通过栈中reference数据操作堆上具体对象,reference类型是一个指向对象引用,没有定义这个引用通过什么方式去定位和访问堆中对象。
主流分两种:
1.句柄:堆中将划分一块内存作为句柄池,reference中存储对象的句柄地址。
             句柄中包含对象实例数据指针和对象类型数据指针 。(最大好处垃圾回收是对象被移动了reference本身不变)

2.直接指针:reference中存储对象实例数据,和对象类型数据指针。
                    如果只访问对象本身就节省一次间接访问的开销。(最大好处访问量大的时候速度快节省指针定位的时间开销)

HotSpot采用的是直接指针方式。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

技术分子

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值