简介
咱们晓得JIT会在JVM运行过程中,对热点代码进行优化,传说天然是传说,明天咱们通过一个简略的例子来具体分析一下JIT到底是怎么进行优化的。
一个简略的例子
说干就干,咱们先筹备一个非常简单的例子:
public class AddTest { static int a = 1; static int b = 2; static int c = 3; public static void main(String[] args) { for (int i = 0; i < 100000; i++) { add(); } } private static void add() { a = b + 1; b = c + 2; c = a + 3; }}
这个例子中咱们定义了三个类变量,而后通过一个add办法对其中的变量进行累加。
而后在main办法中对add办法调用10000次。调用这么屡次,次要是为了保障add成为热点代码,从而应用JIT进行编译。
应用jitWatch进行剖析
之前提到了JIT剖析的神器jitWatch,明天咱们来应用jitWatch来剖析下面的代码。
从jitWatch的github中下载源码,运行mvn exec:java即可开启jitWatch之旅。
关上sandbox,抉择咱们编写的类文件。点击运行即可。
有不相熟jitWatch的敌人能够参考我之前写的文章:
JIT的Profile神器JITWatch
而后咱们到了上面相熟的界面:
界面分为三局部,右边是源代码,两头是字节码,最左边是JIT编译的汇编代码。
剖析字节码
咱们剖析下add办法生成的字节码:
0: getstatic #13 // Field b:I 3: iconst_1 4: iadd 5: putstatic #17 // Field a:I 8: getstatic #20 // Field c:I11: iconst_2 12: iadd 13: putstatic #13 // Field b:I16: getstatic #17 // Field a:I19: iconst_3 20: iadd 21: putstatic #20 // Field c:I24: return
咱们能够看到字节码和java源代码是一一对应的。
比方add办法的第一行:
a = b + 1;
相应的字节码是这样的:
0: getstatic #13 // Field b:I 3: iconst_1 4: iadd 5: putstatic #17 // Field a:I
首先通过getstatic拿到字段b的值。而后调用iconst_1,将1加载。接着调用iadd把1和b相加。最初将生成的值应用putstatic赋值给a。
字节码和源代码一一对应,齐全没有问题。
剖析汇编代码
那么JIT生成的汇编代码是不是也和java代码统一呢?咱们再来看一下生成的汇编代码。
从图片咱们能够看出,生成的汇编代码能够分为办法初始化,代码逻辑区,多线程同步,地址和cache line对齐,异样解决,返优化等几个局部。
这里咱们次要关注一下代码逻辑区:
从图上我做的标记能够看出,汇编中执行的逻辑是
b=c+2, a =b+1和c=b+4。
不光执行程序发送了变动(重排序),执行逻辑也进行了优化。
大家可能留神到汇编语言中有这样几个不太明确的代码:
0x78(%r10)0x74(%r10)0x70(%r10)
通过第二行的注解,咱们晓得r10存储的是AddTest这个对象,而0x70,0x74和0x78是AddTest中的偏移量,用来定位类变量a,b,c。
总结
从下面的例子能够晓得,JIT会对代码进行优化,所以最好的方法是不要本人在java代码中做一些你认为是优化的优化,因为这样可能让JIT在优化的时候变得困惑。从而限度了代码优化的力度。
最初,JIT是一个十分弱小的工具。心愿大家可能喜爱。
本文作者:flydean程序那些事本文链接:http://www.flydean.com/jvm-jitwatch-assembly-indetail/
本文起源:flydean的博客
欢送关注我的公众号:程序那些事,更多精彩等着您!