JVM 给对象分配内存空间

  • 指针碰撞
  • 空闲列表
  • TLAB

为对象分配空间的任务实际上便等同于把一块确定大小的内存块从Java堆中划分出来。

指针碰撞:(Bump The Pointer) 堆的内存是绝对规整的,内存主要分为两部分,所有使用过的内存被放在一边,空闲的内存被放在另一边,中间放着一个指针作为分界点的指示器,分配空间的时候,仅仅把指针向空闲方向挪动一段与对象大小相等的距离。

实现简单,多线程下,效率不高,CAS+失败重试的形式完成的。

空闲列表:(Free List) 如果堆的内存并不是规整的,已被使用的内存和空闲的内存相互交错在一起,那就没有办法简单地进行指针碰撞了,虚拟机就必须维护一个列表,记录上哪些内存块是可用的,在分配的时候从列表中找到一块足够大的空间划分给对象实例,并更新列表上的记录。

有空间碎片的问题

两种方式的选择由 Java 堆是否规整决定,Java 堆是否规整是由选择的垃圾收集器是否具有压缩整理能力决定的。

并发情况下,线程不安全:在给A分配内存,指针没有来得及修改,对象B又使用了原来的指针分配内存,解决方式:

  • CAS+失败重试保证操作的原子性
  • TLAB: 每个线程在Java堆中预先分配一小块内存,称为本地线程分配缓冲,哪个线程要分配内存,就在哪个线程的本地缓冲区中分配,只有本地缓冲区用完了,分配新的缓存区时才需要同步锁定。

你可能感兴趣的:(jvm,java,开发语言)