转载

JIT即时编译器

静态编译,动态编译,动态解释

  • 静态编译: c++程序运行前,将代码编译成机器码,也叫事前编译
  • 动态编译: JIT编译器,程序运行时,将java字节码编译成机器码
  • 动态解释: 程序运行过程中,逐行将字节码解释成机器码

为什么需要JIT编译器和解释器并存?

  • 编译器,适用于热点代码,不用每次都逐行解释,但是编译后的结果需要放到方法区中,占用内存。
  • 解释器,不用占用内存,但是每次都编译,效率低
  • 两者结合,各取所长

什么样的代码会被编译?

  • 多长运行的方法
  • 多次执行的循环体

如何判断时热点代码?

  • 基于计数器采样,方法调用计数器(方法区)和回边计数器(循环体中)
  • 执行的次数超过10000次。clien运行的时候 超过1500次
  • c1和c2的区别,client compiler 注重编译速度,server compiler 注重编译质量

jit编译器的优化

  • 消除公共子表达式

    int d = (c b) 12+a+(a+b*c); //将cb 用E替换

    int d = E 13+a 2;

  • 方法内联

    private int add4(int x1, int x2, int x3, int x4) {
    return add2(x1, x2) + add2(x3, x4);
    }
    private int add2(int x1, int x2) {
    return x1 + x2; }
    //编译器优化后
    private int add4(int x1, int x2, int x3, int x4) {
    return x1 + x2 + x3 + x4; }
  • 逃逸分析 -XX:+DoEscapeAnalysis jdk 1.7 以后默认开启了逃逸分析
  • 什么是逃逸分析

    • jit即时编译器在优化代码过程中,会分析方法内中的变量是否会被外部调用到

      • 静态变量肯定会被外部访问到
      • 将变量当作返回值返回的时候,也会被访问到
      • this 引用。也会被其他方法使用到。
      • 以上就是变量逃逸出去了
    • 对象栈上内存分配

      对象不会逃逸到方法外的时候,将对象从堆内存分配改为栈内存分配,减少堆内存占用和gc

    • 标量替换

      int a = user.a

      int b= user.b

      这个时候,user 是在堆内存分配的。我们可以直接在栈中用两个标量来替换掉

      对象不被外界访问的时候,拆解成若干个成员变量。减少gc

    • 消除同步锁 (-XX:-EliminateLocks )

      只有一个线程在访问的时候,没有必要获取锁,消除掉同步锁。

      public static String getString(String s1, String s2) {
      StringBuffer sb = new StringBuffer();
      sb.append(s1);
      sb.append(s2);
      return sb.toString();
       }
原文  https://segmentfault.com/a/1190000020883615
正文到此结束
Loading...