文档章节

JVM系列之:通过一个例子分析JIT的汇编代码

flydean
 flydean
发布于 07/31 10:08
字数 1038
阅读 4K
收藏 3

行业解决方案、产品招募中!想赚钱就来传!>>>

简介

我们知道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: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的博客

欢迎关注我的公众号:程序那些事,更多精彩等着您!

flydean
粉丝 21
博文 504
码字总数 641192
作品 0
深圳
私信 提问
加载中
请先登录后再评论。
Netty那点事(三)Channel与Pipeline

Channel是理解和使用Netty的核心。Channel的涉及内容较多,这里我使用由浅入深的介绍方法。在这篇文章中,我们主要介绍Channel部分中Pipeline实现机制。为了避免枯燥,借用一下《盗梦空间》的...

黄亿华
2013/11/24
2W
22
访问安全控制解决方案

本文是《轻量级 Java Web 框架架构设计》的系列博文。 今天想和大家简单的分享一下,在 Smart 中是如何做到访问安全控制的。也就是说,当没有登录或 Session 过期时所做的操作,会自动退回到...

黄勇
2013/11/03
3.4K
6
Flappy Bird(安卓版)逆向分析(一)

更改每过一关的增长分数 反编译的步骤就不介绍了,我们直接来看反编译得到的文件夹 方法1:在smali目录下,我们看到org/andengine/,可以知晓游戏是由andengine引擎开发的。打开/res/raw/at...

enimey
2014/03/04
5.9K
18
我的架构演化笔记 功能1: 基本的用户注册

“咚咚”,一阵急促的敲门声, 我从睡梦中惊醒,我靠,这才几点,谁这么早, 开门一看,原来我的小表弟放暑假了,来南京玩,顺便说跟我后面学习一个网站是怎么做出来的。 于是有了下面的一段...

强子哥哥
2014/05/31
976
3
Swift百万线程攻破单例(Singleton)模式

一、不安全的单例实现 在上一篇文章我们给出了单例的设计模式,直接给出了线程安全的实现方法。单例的实现有多种方法,如下面: class SwiftSingleton { } 这段代码的实现,在shared中进行条...

一叶博客
2014/06/20
3.3K
16

没有更多内容

加载失败,请刷新页面

加载更多

鼠年吉祥,新年快乐

今天是大年初一,很高兴在过去一年中有您的陪伴,希望大家在新的一年中平安健康,一切顺利,加油。 邓飞 202001250539 于后园爷爷家 本文分享自微信公众号 - 育种数据分析之放飞自我(R-bre...

育种数据分析之放飞自
01/25
0
0
不烧脑、不耗时、全免费,带你0基础学Python

文末有福利 Python是人工智能的未来。 最近,电气和电子工程师协会( IEEE)发布了顶级编程语言交互排行榜:Python高居首位。 而且随着大数据和人工智能的发展,Python受到了越来越多程序员的...

kunjian
今天
0
0
R语言入门系列之一

写在前面 计算机语言的学习并不困难,关键是一定要由浅入深的实际操作练习。也许最开始的比较简单,学习者一带而过没有实际操作,之后的进一步学习很可能会陷入不知所云的困境,实际操作所带...

SYSU星空
2019/02/17
0
0
Istio-本地运行

概述 基于上一篇 Istio1.6-二进制编译和本地运行 但集中在 pilot-discovery 和 envoy(pilot-agent 大部分功能仅作为 envoy 的 watchdog,略过) NOTE: 以下的描述,相对路径都基于目录 /g...

深蓝苹果
17分钟前
9
0
基于Linux、C、JSON、Socket的编程实例(附代码)

点击上方「嵌入式大杂烩」,选择「置顶公众号」第一时间阅读编程笔记! 一、前言 之前在学习socket编程的时候有分享一个基于控制台的简易天气客户端的实现,当时提供的是window下的代码,最近...

学以解忧
2019/10/29
0
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部