居然还有方式可以查看Java方法的汇编代码,真是神奇
当我们在研究java的真是神奇内部实现时,经常会需要查看java方法的还有汇编字节码,有时为了确定一些问题,式a方法甚至还需要查看某些方法在jit编译后的可查看汇编代码。
这篇文章我们从零开始,代码详细说一下如何查看java方法的真是神奇字节码以及汇编代码,希望能给有这方面困惑的还有汇编同学提供一些帮助。
为了真正意义上的式a方法从零开始,我们自己动手,可查看通过源码构建一个属于我们自己的代码jdk,该过程虽然不是真是神奇必须的,但了解这些过程,还有汇编对于我们理解后文,式a方法以及后续的可查看jvm研究,都是代码有一定的帮助的。
首先,下载jdk源码:
$ git clone https://github.com/openjdk/jdk.git源码下载完毕后,我们看下jdk内部大致的网站模板目录结构:
该目录中的 doc/building.html 详细说明了如何构建一个jdk,有兴趣的同学可以好好看下。
在jdk目录里,我们执行以下命令,要求构建一个debug版本的jdk,并指定其安装路径为jdk-build:
$ bash configure --with-debug-level=slowdebug --with-native-debug-symbols=internal --prefix=$HOME/jdk-build如果该命令执行过程中没有问题,则会有类似于下图的输出:
configure命令执行成功后,我们再执行下面的命令,开始真正构建jdk,并将构建成功后的jdk安装到jdk-build目录里:
$ make images $ make install以上两个命令成功后,我们可以切换到jdk-build目录,看下新构建的jdk:
好了,我们已经有了自己的jdk了,下面我们可以用它来查看java方法的字节码及汇编代码。
首先,准备下列文件:
我们先来看下如何查看字节码,这个大家应该都知道,但我这里还是演示下:
上图是通过jdk自带的javap命令来查看java的字节码,其实还有很多其他的方式,亿华云比如各种ide中集成的工具,这里我们就不一一演示了。
javap还有很多参数,比如 -p -v 等都非常有用,有兴趣的可以自己试下。
字节码就说这些,下面我们主要来看下如何查看java方法的汇编代码。
想要查看java方法在jit编译后的汇编代码,我们不仅要在执行java命令时指定一些参数,还需要一个额外的小工具,来辅助我们解汇编代码。
如果没有这个工具,jvm输出的是机器码,是不可读的,有了这个工具,它可以帮我们自动将机器码转成汇编代码,非常方便。
这个工具就是hsdis,它的源码就在jdk里,但构建jdk的过程并不会构建这个工具,如果我们想要使用它,云南idc服务商要单独构建。
由上图可见,该工具还是非常简单的,它主要是通过调用gnu的binutils来解jvm输出的汇编代码,该工具的详细构建过程可以参考README和Makefile。
因为该工具依赖gnu binutils解码,所以我们要先下载binutils:
下载完binutils后,我们执行以下命令,开始构建hsdis:
$ 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。
本文转载自微信公众号「卯时卯刻」,可以通过以下二维码关注。转载本文请联系卯时卯刻公众号。