JVM系列之:通过一个例子分析JIT的汇编代码
简介
我们知道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编译的汇编代码。
分析字节码
我们分析下add方法生成的字节码:
0: getstatic #13 // Field b:I 3: iconst_1 4: iadd 5: putstatic #17 // Field a:I 8: getstatic #20 // Field c:I 11: iconst_2 12: iadd 13: putstatic #13 // Field b:I 16: getstatic #17 // Field a:I 19: iconst_3 20: iadd 21: putstatic #20 // Field c:I 24: 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的博客
欢迎关注我的公众号:程序那些事,更多精彩等着您!

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
C++模板沉思录(上)
**花下猫语:**在我们读者群里,最近出现了比较多关于 C++ 的讨论,还兴起了一股学习 C++ 的风气。樱雨楼小姐姐对 C++ 的模板深有研究,系统地梳理成了一篇近 4 万字的文章!本文是上篇,分享给大家~ 樱雨楼 | 原创作者 豌豆花下猫 | 编辑 0 论抽象——前言 故事要从一个看起来非常简单的功能开始: 请计算两个数的和。 如果你对Python很熟悉,你一定会觉得:“哇!这太简单了!”,然后写出以下代码: def Plus(lhs, rhs): return lhs + rhs 那么,C语言又如何呢?你需要面对这样的问题: /* 这里写什么?*/ Plus(/* 这里写什么?*/ lhs, /* 这里写什么?*/ rhs) { return lhs + rhs; } 也许你很快就能想到以下解法中的一些或全部: 硬编码为某个特定类型: int Plus(int lhs, int rhs) { return lhs + rhs; } 显然,这不是一个好的方案。因为这样的Plus函数接口强行的要求两个实参以及返回值的类型都必须是int,或是能够发生隐式类型转换到int的类型。此时,如...
- 下一篇
PHP 开发框架 ebcms-framework v1.0.9 发布
写在最前 很多人好奇,全球那么多php框架,为什么还要重复造轮子? 本人从事php开发超过5年,用过很多的框架,thinkphp、laravel、zendframework...,也用过很多成熟的基于php开发的产品,比如wordpress、discuz、phpcms、dedecms...,但总是这里或者哪里不让我如意的地方。 我心中理想的框架应该具备: 标准化:常用的操作、通用的类库,应该遵守统一的规范或接口 应用生态:应该有很好的应用分享机制 国际化:世界各地的开发者都能够无缝参与进来 于是ebcms/framework出来了 项目特点 1.严格遵守PSR规范 本项目严格遵守PSR规范,消息、日志、缓存、容器、事件、代码规范... 遵守PSR规范带来的直接好处是即刻拥有了composer上的海量的生态资源,并且更容易让其他php开发者无缝参与进来。 2.先进的应用生态理念 本项目原生理念就是为应用分享,基于composer优秀的包依赖机制,您开发的APP可以通过composer被世界各地的人使用,您也可以使用别人开发的APP。 3.极简内核 得益于遵守PSR规范,我们的内核极度简单...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- SpringBoot2整合Thymeleaf,官方推荐html解决方案
- Jdk安装(Linux,MacOS,Windows),包含三大操作系统的最全安装
- CentOS7设置SWAP分区,小内存服务器的救世主
- SpringBoot2初体验,简单认识spring boot2并且搭建基础工程
- SpringBoot2全家桶,快速入门学习开发网站教程
- Docker安装Oracle12C,快速搭建Oracle学习环境
- SpringBoot2整合MyBatis,连接MySql数据库做增删改查操作
- Eclipse初始化配置,告别卡顿、闪退、编译时间过长
- MySQL8.0.19开启GTID主从同步CentOS8
- CentOS8,CentOS7,CentOS6编译安装Redis5.0.7