当我们在研究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。
好,就这些,写文章不易,如果可以的话,帮忙转发下或给个在看,谢谢。