JVM对象分配之栈上分配 & TLAB分配

Java对象分配流程java

栈上分配技术:

是java虚拟机提供的一项优化技术,它的基本思想是,对于那些线程私有对象(指不可能被其余线程访问的对象)能够将它们打散分配在栈上,而不是分配在堆上。缓存

好处: 分配在栈上能够结束后自行销毁,不须要垃圾回收器介入,从而提升系统的性能。性能优化

局限性: 栈空间小,对于大对象没法实现栈上分配。bash

基础:栈上分配依赖于逃逸分析和标量替换。markdown

逃逸分析:

栈上分配的一个技术基础是进行逃逸分析。目的是判断对象的做用域是否有可能逃逸出逃逸体。多线程

虚拟机会进行逃逸分析,判断线程内私有对象是否有可能被其余线程访问,致使逃逸,而后虚拟机就会根据是否可能会逃逸将其分配在栈上,或者堆中。函数

只有在server模式下,才能开启逃逸分析。 以下示例: oop

参数: -XX:+DoEscapeAnalysis 是开启逃逸分析。 -XX:+EliminateAllocations 是开启标杆替换,容许将对象打散分配到栈上,默认就是打开的。

代码:性能

//user的做用域超出了函数setUser的范围,是逃逸对象
//当函数结束调用时,不会自行销毁user
private User user;
public void setUser(){
   user = new User();
   user.setId(1);
   user.setName("blueStarWei");
}

//u只在函数内部生效,不是逃逸对象
//当函数调用结束,会自行销毁对象u
public void createUser(){
   User u = new User();
   u.setId(2);
   u.setName("JVM");
}
复制代码

栈上示例分配:<来自实战java虚拟机>优化

public class AllotOnStack {

   public static class{
       public int id=0;
       public String name="";
   }

   public static void main(String[] args) {
       long start = System.currentTimeMillis();
       for (int i = 0; i < 100000000; i++) {
           alloc();
       }
       long end = System.currentTimeMillis();
       System.out.println(end - start);
   }

   private static void alloc() {
       User user = new User();
       user.setId(1);
       user.setName("zengxinyao");
   }
}
复制代码

上述代码调用了1亿次alloc(),若是是分配到堆上,大概须要1.5GB的堆空间,若是堆空间小于该值,必然会触发GC。

使用以下参数运行,发现不会触发GC

-server -Xmx15m -Xms15m -XX:+DoEscapeAnalysis -XX:+PrintGC -XX:-UseTLAB -XX:+EliminateAllocations
复制代码

使用以下参数(任意一行)运行,会发现触大量GC

//不使用逃逸分析
-server -Xmx15m -Xms15m -XX:-DoEscapeAnalysis -XX:+PrintGC -XX:-UseTLAB -XX:+EliminateAllocations

//不使用标量替换
-server -Xmx15m -Xms15m -XX:+DoEscapeAnalysis -XX:+PrintGC -XX:-UseTLAB -XX:-EliminateAllocations
复制代码

能够得出: 栈上分配依赖于逃逸分析和标量替换

JVM参数解析

TLAB 分配

TLAB,全称Thread Local Allocation Buffer,即:线程本地分配缓存。这是一块线程专用的内存分配区域。TLAB占用的是eden区的空间。在TLAB启用的状况下(默认开启),JVM会为每个线程分配一块TLAB区域。

为何须要TLAB?

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

局限性:TLAB空间通常不会太大(占用eden区),因此大对象没法进行TLAB分配,只能直接分配到堆上.

分配策略:

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

  此时,虚拟机有两种选择:第一,废弃当前的TLAB(会浪费20KB的空3.4 间);第二,将这个30KB的对象直接分配到堆上,保留当前TLAB(当有小于20KB的对象请求TLAB分配时能够直接使用该TLAB区域)。

  JVM选择的策略是:在虚拟机内部维护一个叫refill_waste的值,当请求对象大于refill_waste时,会选择在堆中分配,反之,则会废弃当前TLAB,新建TLAB来分配新对象。

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

取自: 《实战Java虚拟机 - JVM故障诊断与性能优化》 栈上分配、TLAB : blog.csdn.net/yangsnow_ra…