前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >漫画:奇怪,为什么在Java中 2*(i*i) 比 2*i*i 快?

漫画:奇怪,为什么在Java中 2*(i*i) 比 2*i*i 快?

作者头像
纯洁的微笑
发布2019-05-06 15:45:24
7860
发布2019-05-06 15:45:24
举报
文章被收录于专栏:纯洁的微笑

既然我设计的两只小萌宠出场了,也该它们的粑粑出场了,今天这篇文章,我们通过一个故事来深入聊聊 Java 编译背后的秘密。

东哥说这段代码来自于 Stackoverflow,真正去逛这个网站的同学还是很少),如下。

代码语言:javascript
复制
public static void main(String[] args) {
    long startTime = System.nanoTime();
    int n = 0;
    for (int i = 0; i < 1000000000; i++) {
        n += 2 * i * i;
    }
    System.out.println((double) (System.nanoTime() - startTime) / 1000000000 + " s");
    System.out.println("n = " + n);
}

代码很简单吧,它的执行时间大概在 0.60s ~ 0.65s。但是如果你把 2*i*i 替换成 2*(i*i),执行时间大概在 0.50s ~ 0.55s。

对这段程序的两个版本分别执行 15 次,得到的结果如下。

我们可以看出 2*(i*i) 比 2*i*i 快 。

我们来分别查看它的字节码,这里东哥给我推荐了一款好用的 IDEA 插件,叫做 jclasslib bytecode viewer。

2*i*i 的字节码如下。

2*(i*i) 的字节码如下。

我们可以发现除了字节码顺序不同外,没有其它异常,下一步该怎么办呢?

这里东哥给我推荐一款开源的工具,叫做 jitwatch,它可以查看查看分析HotSpot JIT compiler 生成的汇编代码,关于它的安装方法可以通过谷歌查阅。

查看是否安装成功?可以用下面的命令。

如何查看汇编代码?

运行时,添加参数 -XX:+PrintAssembly。

好了,那我们来分别看看它们的汇编代码,由于汇编代码很多,我这里就不贴出来了,各位同学可以去运行分析一下。

我这里就说一下结论,通过对比分析,我们会发现,2*i*i 进行了大量的堆栈操作,因此,需要保存大量的中间结果;而 2*(i*i) 只有少量的堆栈操作。

显而易见,2*(i*i) 比 2*i*i 快是由于 JIT 优化的结果。

-END-

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2019-01-28,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 纯洁的微笑 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档