对象 替换_JVM 对象分配过程
对象分配过程
- 1)依据逃逸分析,判断是否能栈上分配?
- 如果可以,使用标量替换方式,把对象分配到
VM Stack
中。如果 线程销毁或方法调用结束后,自动销毁,不需要 GC 回收器 介入。 - 否则,继续下一步。
- 如果可以,使用标量替换方式,把对象分配到
- 2)判断是否大对象?
- 如果是,直接分配到堆上
Old Generation
老年代上。如果对象变为垃圾后,由老年代GC 收集器(比如 Parallel Old, CMS, G1)回收。 - 否则,继续下一步。
- 如果是,直接分配到堆上
- 3)判断是否可以在
TLAB
中分配?- 如果是,在
TLAB
中分配堆上Eden
区。 - 否则,在
TLAB
外堆上的Eden
区分配。
- 如果是,在
栈上分配
本质上是JVM提供的一个优化技术。
- 基本思想:将线程私有的对象打散分配在栈
VM Stack
上 - 优点:
- 可以在函数调用结束后自行销毁对象,不需要垃圾回收器的介入,有效避免垃圾回收带来的负面影响
- 栈上分配速度快,提高系统性能
- 局限性:
- 栈空间小,对于大对象无法实现栈上分配
- 技术基础:
逃逸分析
、标量替换
什么是逃逸分析?
关于 Java 逃逸分析的定义:
逃逸分析(Escape Analysis)简单来讲就是,Java Hotspot 虚拟机可以分析新创建对象的使用范围,并决定是否在 Java 堆上分配内存的一项技术。
逃逸分析的 JVM 参数如下:
- 开启逃逸分析:
-XX:+DoEscapeAnalysis
- 关闭逃逸分析:
-XX:-DoEscapeAnalysis
- 显示分析结果:
-XX:+PrintEscapeAnalysis
逃逸分析技术在 Java SE 6u23+ 开始支持,并默认设置为启用状态,可以不用额外加这个参数。
逃逸分析优化
针对上面第三点,当一个对象没有逃逸时,可以得到以下几个虚拟机的优化。
1) 锁消除
我们知道线程同步锁是非常牺牲性能的,当编译器确定当前对象只有当前线程使用,那么就会移除该对象的同步锁。
例如,StringBuffer 和 Vector 都是用 synchronized 修饰线程安全的,但大部分情况下,它们都只是在当前线程中用到,这样编译器就会优化移除掉这些锁操作。
锁消除的 JVM 参数如下:
- 开启锁消除:
-XX:+EliminateLocks
- 关闭锁消除:
-XX:-EliminateLocks
锁消除在 JDK8 中都是默认开启的,并且锁消除都要建立在逃逸分析的基础上。
2) 标量替换
首先要明白标量和聚合量,基础类型和对象的引用可以理解为标量,它们不能被进一步分解。而能被进一步分解的量就是聚合量,比如:对象。
对象是聚合量,它又可以被进一步分解成标量,将其成员变量分解为分散的变量,这就叫做标量替换
。
这样,如果一个对象没有发生逃逸,那压根就不用创建它,只会在栈或者寄存器上创建它用到的成员标量,节省了内存空间,也提升了应用程序性能。
标量替换的 JVM 参数如下:
- 开启标量替换:
-XX:+EliminateAllocations
- 关闭标量替换:
-XX:-EliminateAllocations
- 显示标量替换详情:
-XX:+PrintEliminateAllocations
标量替换同样在 JDK8 中都是默认开启的,并且都要建立在逃逸分析的基础上。
3) 栈上分配
当对象没有发生逃逸时,该对象就可以通过标量替换分解成成员标量分配在栈内存中,和方法的生命周期一致,随着栈帧出栈时销毁,减少了 GC 压力,提高了应用程序性能。
示例代码
import java.time.Instant;
/*** 栈上分配,依赖于逃逸分析和标量替换** @author Sven Augustus*/
public class TestTLAB {// private static User u;/*** 一个User对象的大小:markdown 8 + class pointer 4 + int 4 + string (oops) 4 + padding 4 = 24B <br> 如果分配 100_000_000 个,则需要* 2400_000_000 字节, 约 2.24 GB。*/static class User {private int id;private String name;public User(int id, String name) {this.id = id;this.name = name;}}private static void alloc() {User u = new User(1, "SvenAugustus");// u = new User(1, "SvenAugustus");}public static void main(String[] args) throws InterruptedException {long start = Instant.now().toEpochMilli();for (int i = 0; i < 100_000_000; i++) {alloc();}System.out.println(Instant.now().toEpochMilli() - start);}
}
上述代码调用了1亿次alloc(),如果是分配到堆上,大概需要 2.2 GB的堆空间,如果堆空间小于该值,必然会触发GC。
使用如下VM参数运行,发现不会触发GC:
-server -Xmx15m -Xms15m -XX:+DoEscapeAnalysis -XX:+PrintGC -XX:-UseTLAB -XX:+EliminateAllocations
使用如下参数(任意一行)运行,会发现触大量 GC:
//不使用逃逸分析
-server -Xmx15m -Xms15m -XX:+PrintGCDetails -XX:-UseTLAB -XX:-DoEscapeAnalysis -XX:+EliminateAllocations
//不使用标量替换
-server -Xmx15m -Xms15m -XX:+PrintGCDetails -XX:-UseTLAB -XX:+DoEscapeAnalysis -XX:-EliminateAllocations
TLAB 分配
TLAB,全称Thread Local Allocation Buffer, 即:线程本地分配缓存。这是一块线程专用的内存分配区域。
TLAB占用的是eden区的空间。
在TLAB启用的情况下(默认开启),JVM会为每一个线程分配一块TLAB区域。
为什么需要TLAB?
这是为了加速对象的分配。
由于对象一般分配在堆上,而堆是线程共用的,因此可能会有多个线程在堆上申请空间,而每一次的对象分配都必须线程同步,会使分配的效率下降。
考虑到对象分配几乎是Java中最常用的操作,因此JVM使用了TLAB这样的线程专有区域来避免多线程冲突,提高对象分配的效率。
- 局限性: TLAB空间一般不会太大(占用eden区),所以大对象无法进行
TLAB
分配,只能直接分配到堆Heap
上。
大对象
大对象的 JVM 参数如下:
- 大对象到底多大:
-XX:PreTenureSizeThreshold=n
(仅适用于DefNew
/ParNew
新生代垃圾回收器 ) https://bugs.openjdk.java.net/browse/JDK-8050209 G1
回收器的大对象判断,则依据Region
的大小(-XX:G1HeapRegionSize
)来判断,如果对象大于Region
50%以上,就判断为大对象Humongous Object
。
by Sven Augustus https://my.oschina.net/langxSpirit
对象 替换_JVM 对象分配过程相关推荐
- 对象布局(JOL)、分配过程以及访问定位
一.对象布局内存结构:对象头.实例数据.填充补齐(非必须) 无继承关系: 有继承关系: 1.对象头:以32位操作系统为例 对象头形式: 1)普通对象:8个字节(64位系统,不开启压缩指针是16个字节, ...
- JVM成神路对象内存布局、分配过程、从生至死历程、强弱软件引用
引言 对象实例的角度,阐述一个Java对象从生到死的历程.Java对象在内存中的布局以及对象引用类型. 一.Java对象在内存中的布局 Java源代码中,使用new关键字创建出的对象实例,我们都知道在 ...
- (五)JVM成神路之对象内存布局、分配过程、从生至死历程、强弱软虚引用全面剖析
引言 在上篇文章中曾详细谈到了JVM的内存区域,其中也曾提及了:Java程序运行过程中,绝大部分创建的对象都会被分配在堆空间内.而本篇文章则会站在对象实例的角度,阐述一个Java对象从生到死的历程.J ...
- ART对象内存分配过程解析(上)——内存分配的准备阶段(Android 8.1)
注:本文基于Android 8.1进行分析. ART对象分配过程解析--内存分配的准备阶段 本章我们将分析Android 8.1中ART虚拟机的对象创建时内存分配过程的分析.本节将介绍内存分配相关的环 ...
- JVM学习笔记之-堆,年轻代与老年代,对象分配过程,Minor GC、Major GC、Full GC,堆内存大小与OOM,堆空间分代,内存分配策略,对象分配内存,小结堆空间,逃逸分析,常用调优工具
堆的核心概述 概述 一个JVM实例只存在一个堆内存,堆也是Java内存管理的核心区域.Java堆区在JVM 启动的时候即被创建,其空间大小也就确定了.是JVM管理的最大一块内存空间. 堆内存的大小是可 ...
- JVM---堆(对象分配过程)
堆-对象分配过程 堆中TLAB为对象分配内存 堆-逃逸分析与代码优化 概念 为新对象分配内存是一件非常严谨和复杂的任务,JVM的设计者们不仅需要考虑内存如何分配.在哪里分配等问题,并且由于内存分配算法 ...
- 独占设备的分配与回收_灵魂拷问:Java对象的内存分配过程是如何保证线程安全的?...
点击上方"linkoffer", 选择关注公众号高薪职位第一时间送达 作者 l Hollis JVM内存结构,是很重要的知识,相信每一个静心准备过面试的程序员都可以清楚的把堆.栈. ...
- 原创 | 灵魂拷问:Java对象的内存分配过程是如何保证线程安全的?
△Hollis, 一个对Coding有着独特追求的人△ 这是Hollis的第 222 篇原创分享 作者 l Hollis 来源 l Hollis(ID:hollischuang) JVM内存结构,是很 ...
- G1垃圾收集器之对象分配过程
本文来说下G1垃圾收集器之对象分配过程 文章目录 概述 小对象 大对象 本文小结 概述 G1的年轻代由eden region 和 survivor region 两部分组成,新建的对象(除了巨型对象) ...
- JVM源码简析(楔子)-对象内存分配过程和PS回收器中YGC触发FGC的现象
前言 想要搞明白Java对象内存申请过程的原因,是因为第一次接触线上GC日志的时候,发现了一些很奇怪的现象,就是young gc触发了full gc.为了搞清楚这个现象,得先要来个测试去复现. 复现现 ...
最新文章
- 适合初学者的java书籍
- 【Android】Touch事件分发
- 过程工程中的计算机应用基础,CDIO工程教育培养模式在《计算机应用基础》课中的应用...
- HTML+CSS公司培训(一)高手请飘过
- 现代软件工程系列 学生的精彩文章 (4) 为用户服务
- node中使用短信验证功能(阿里云为例)
- NYOJ题目66-分数拆分
- Java—String转化为JsonObject
- 三层交换(VLAN间路由)
- 未来互联网+大数据时代
- openwrt mesh网络设置
- jquery 报错提示Uncaught TypeError: $ is not a function
- 奔驰S400商务型升级前排通风座椅系统,夏天必备的功能
- Office快捷键大全之二(Excel快捷键)
- ubuntu 使用LVM修改分区大小后开机报错的解决办法
- 2020北航计算机夏令营
- xlwt/xlrd库的区别
- 联想计算机boss设置,联想电脑bios设置图解教程
- 服务器项目报备什么意思,报备项目是什么意思
- centOS和宝塔linux面板详细使用