居然还有方式可以查看Java方法的汇编代码,真是神奇。

2020-10-30 11:48:05 浏览数 (1)

当我们在研究java的内部实现时,经常会需要查看java方法的字节码,有时为了确定一些问题,甚至还需要查看某些方法在jit编译后的汇编代码。

这篇文章我们从零开始,详细说一下如何查看java方法的字节码以及汇编代码,希望能给有这方面困惑的同学提供一些帮助。

为了真正意义上的从零开始,我们自己动手,通过源码构建一个属于我们自己的jdk,该过程虽然不是必须的,但了解这些过程,对于我们理解后文,以及后续的jvm研究,都是有一定的帮助的。

首先,下载jdk源码:

代码语言:javascript复制
$ git clone https://github.com/openjdk/jdk.git

源码下载完毕后,我们看下jdk内部大致的目录结构:

该目录中的 doc/building.html 详细说明了如何构建一个jdk,有兴趣的同学可以好好看下。

在jdk目录里,我们执行以下命令,要求构建一个debug版本的jdk,并指定其安装路径为jdk-build:

代码语言:javascript复制
$ bash configure --with-debug-level=slowdebug --with-native-debug-symbols=internal --prefix=$HOME/jdk-build

如果该命令执行过程中没有问题,则会有类似于下图的输出:

configure命令执行成功后,我们再执行下面的命令,开始真正构建jdk,并将构建成功后的jdk安装到jdk-build目录里:

代码语言:javascript复制
$ make images
$ make install

以上两个命令成功后,我们可以切换到jdk-build目录,看下新构建的jdk:

好了,我们已经有了自己的jdk了,下面我们可以用它来查看java方法的字节码及汇编代码。

首先,准备下列文件:

我们先来看下如何查看字节码,这个大家应该都知道,但我这里还是演示下:

上图是通过jdk自带的javap命令来查看java的字节码,其实还有很多其他的方式,比如各种ide中集成的工具,这里我们就不一一演示了。

javap还有很多参数,比如 -p -v 等都非常有用,有兴趣的可以自己试下。

字节码就说这些,下面我们主要来看下如何查看java方法的汇编代码。

想要查看java方法在jit编译后的汇编代码,我们不仅要在执行java命令时指定一些参数,还需要一个额外的小工具,来辅助我们解汇编代码。

如果没有这个工具,jvm输出的是机器码,是不可读的,有了这个工具,它可以帮我们自动将机器码转成汇编代码,非常方便。

这个工具就是hsdis,它的源码就在jdk里,但构建jdk的过程并不会构建这个工具,如果我们想要使用它,要单独构建。

由上图可见,该工具还是非常简单的,它主要是通过调用gnu的binutils来解jvm输出的汇编代码,该工具的详细构建过程可以参考README和Makefile。

因为该工具依赖gnu binutils解码,所以我们要先下载binutils:

下载完binutils后,我们执行以下命令,开始构建hsdis:

代码语言:javascript复制
$ make BINUTILS=binutils-2.35.1 all64

如果没有问题的话,最终会在build/linux-amd64目录下生成一个hsdis-amd64.so文件:

将该文件拷贝到我们之前构建好的jdk里:

好,准备工作已经完成,现在我们可以通过指定一些参数,来查看java方法的汇编代码了。

我们还是用上面那个java类T.java,假设我们想查看方法f1在jit编译后的汇编代码,可以使用下面的命令:

该命令会输出很多内容,而下图中的就是我们想要的:

看到没,真的是汇编,且选中行就是方法f1的相加逻辑。

我们可以通过不同的参数来指定要查看的某个方法或某些方法,我们也可以通过-XX: PrintAssembly参数,来查看所有被jit编译的方法。

有关各参数的使用及意义,请参考以下链接:

https://docs.oracle.com/en/java/javase/15/docs/specs/man/java.html

查看java方法的汇编代码,对于我们理解java的内部实现,是非常有意义的,通过这种方式的辅助,我们可以理解很多文档上难以理解的内容,比如 volatile。


好,就这些,写文章不易,如果可以的话,帮忙转发下或给个在看,谢谢。

0 人点赞