JVM - 栈上分配 & TLAB

Java对象内存分配方式

https://www.cnblogs.com/BlueStarWei/p/9358757.html

(1)从静态存储区域分配。内存在程序编译的时候就已经分配好,这块内存在程序的整个运行期间都存在。例如全局变量,static变量。

(2)在栈上创建。在执行函数时,函数内局部变量的存储单元都可以在栈上创建,函数执行结束时这些存储单元自动释放。栈内存分配运算内置于处理器的指令集中,效率很高,但是分配的内存容量有限。

(3) 从堆上分配,亦称动态内存分配。程序在运行的时候用malloc或new申请任意多少的内存,程序员自己负责在何时用free或delete释放内存。动态内存的生存期由我们决定,使用非常灵活,但问题也最多

Java对象分配流程

栈上分配

Java虚拟机提供的一项优化技术,将线程私有的对象打散分配在栈上;

可以在函数调用结束后自行销毁对象,不需要垃圾回收器的介入,有效避免垃圾回收带来的负面影响

但是栈空间小,对于大对象无法实现栈上分配

栈上分配的一个技术基础是进行 逃逸分析

逃逸分析

判断对象的作用域是否超出函数体[即:判断是否逃逸出函数体]

TLAB 分配

https://blog.csdn.net/yangsnow_rain_wind/article/details/80434323

TLAB,全称Thread Local Allocation Buffer, 即:线程本地分配缓存。这是一块线程专用的内存分配区域。TLAB占用的是eden区的空间,默认设定为占用Eden Space的1%(https://blog.csdn.net/yangzl2008/article/details/43202969)

触发GC的时候,无论是minor GC还是full GC,要收集Eden的时候里面的空间无论是属于某个线程的TLAB还是不属于任何TLAB都一视同仁,把Eden当作一个整体来收集里面的对象

为了加速对象的分配。由于对象一般分配在堆上,而堆是线程共用的,因此可能会有多个线程在堆上申请空间,而每一次的对象分配都必须线程同步,会使分配的效率下降。考虑到对象分配几乎是Java中最常用的操作,因此JVM使用了TLAB这样的线程专有区域来避免多线程冲突,提高对象分配的效率。

TLAB空间一般不会太大(占用eden区),所以大对象无法进行TLAB分配,只能直接分配到堆上

相关 JVM 参数

作用

备注

-XX:+UseTLAB

启用TLAB

默认启用

-XX:TLABRefillWasteFraction

设置允许空间浪费的比例

默认值:64,即:使用1/64的TLAB空间大小作为refill_waste值

-XX:-ResizeTLAB

禁止系统自动调整TLAB大小

 

-XX:TLABSize

指定TLAB大小

单位:B

一个100KB的TLAB区域,如果已经使用了80KB,当需要分配一个30KB的对象时,TLAB是如何分配的呢?

JVM选择的策略是:在虚拟机内部维护一个叫refill_waste的值,当请求对象大于refill_waste时,会选择在堆中分配,

反之,则会废弃当前TLAB,新建TLAB来分配新对象

【默认情况下,TLAB和refill_waste都是会在运行时不断调整的,使系统的运行状态达到最优。】

猜你喜欢

转载自blog.csdn.net/lewee0215/article/details/111240590