JVM源码分析之Java对象的内存分配
instanceKlass
对象计算出需要多大的内存,并调用CollectedHeap
的common_mem_allocate_noinit
方法分配指定大小的内存,实现如下:

从线程的局部缓冲区分配临时内存
TLAB技术是每个线程在Java堆中预先分配了一小块内存,当有对象创建请求内存分配时,就会在该块内存上进行分配,而不需要在Java堆通过同步控制进行内存分配。如果UseTLAB
为真,则使用TLAB技术(Thread-Local Allocation Buffers),将分配工作交由线程自行完成,实现如下:
1、如果线程的局部缓冲区可以分配指定大小的内存,则直接分配;
2、否则执行allocate_from_tlab_slow
在Java堆上进行分配,实现如下:
allocate_new_tlab
从Java堆上重新为线程分配一块局部缓冲区,实现如下:
其中mem_allocate
方法实现从Java堆分配临时内存。
从内存堆中分配临时内存
在内存堆管理器看来,为普通对象分配内存和为某一线程分配一块本地分配缓冲区在本质上都是一样的,这块内存都是临时的,只能从新生代或老年代中进行分配,通过gc策略GenCollectorPolicy::mem_allocate_work
方法进行实现,大概步骤如下:
gch->no_gc_in_progress()
确保当前JVM没有正在进行gc;gc_overhead_limit_was_exceeded
表示当前内存分配操作是否发生了gc,以及gc耗时是否超过设置限制,主要针对一些对延迟敏感的场景,当该参数为true
时,抛出OOM的异常给上层;3、如果分配失败,则执行step 3;
step 2
通过重试机制确保内存能够分配成功:
1、首先在新生代采用无锁的方式尝试分配内存,通过Atomic::cmpxchg_ptr的CAS操作对新生代空闲内存进行同步分配,最终实现如下:

2、参数
first_only
表示当前是否只应该在新生代分配内存,如果新生代的剩余空间不够,则尝试在老年代进行分配;3、依次尝试从内存各个代中分配内存,实现如下:

gc_locker::is_active_and_needs_gc()
为真时,表示当前其它线程已经触发了gc;is_tlab
为真,表示当前线程正在为局部分配缓冲区申请内存;!gch->is_maximal_no_gc()
为真,表示新生代或老年代可以进行内存扩展,扩展完成后,再次尝试从各代中进行分配,实现如下:


VM Thread
的JVM级线程调度执行;2、当操作执行成功并返回时,如果gc锁已被加锁,说明已经由其它线程触发了gc,则继续循环以等待gc完成;
3、否则当前线程等待gc完成,判断gc耗时是否超过设置的gc超时上限,并执行软引用的清除;
4、如果gc超时,则给上层调用返回NULL,让其抛出内存溢出错误;
1、本站所有文档、视频、书籍等资料均由网友分享,本站只负责收集不承担任何技术及版权问题。
2、如本帖侵犯到任何版权问题,请立即告知本站,本站将及时予与删除下载链接并致以最深的歉意。
3、本帖部分内容转载自其它媒体,但并不代表本站赞同其观点和对其真实性负责。
4、一经注册为本站会员,一律视为同意网站规定,本站管理员及版主有权禁止违规用户。
5、蚂蚁编程管理员有权不事先通知发贴者而删除本文。
蚂蚁编程学院 » JVM源码分析之Java对象的内存分配
2、如本帖侵犯到任何版权问题,请立即告知本站,本站将及时予与删除下载链接并致以最深的歉意。
3、本帖部分内容转载自其它媒体,但并不代表本站赞同其观点和对其真实性负责。
4、一经注册为本站会员,一律视为同意网站规定,本站管理员及版主有权禁止违规用户。
5、蚂蚁编程管理员有权不事先通知发贴者而删除本文。
蚂蚁编程学院 » JVM源码分析之Java对象的内存分配