当我们在研究java的真是神奇内部实现时,经常会需要查看java方法的还有汇编字节码,有时为了确定一些问题,式a方法甚至还需要查看某些方法在jit编译后的可查看汇编代码。 这篇文章我们从零开始,代码详细说一下如何查看java方法的真是神奇字节码以及汇编代码,希望能给有这方面困惑的还有汇编同学提供一些帮助。 为了真正意义上的式a方法从零开始,我们自己动手,可查看通过源码构建一个属于我们自己的代码jdk,该过程虽然不是真是神奇必须的,但了解这些过程,还有汇编对于我们理解后文,式a方法以及后续的可查看jvm研究,都是代码有一定的帮助的。 首先,下载jdk源码: 源码下载完毕后,我们看下jdk内部大致的云服务器提供商目录结构: 该目录中的 doc/building.html 详细说明了如何构建一个jdk,有兴趣的同学可以好好看下。 在jdk目录里,我们执行以下命令,要求构建一个debug版本的jdk,并指定其安装路径为jdk-build: 如果该命令执行过程中没有问题,则会有类似于下图的输出: configure命令执行成功后,我们再执行下面的命令,开始真正构建jdk,并将构建成功后的jdk安装到jdk-build目录里: 以上两个命令成功后,我们可以切换到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: 如果没有问题的话,最终会在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。 本文转载自微信公众号「卯时卯刻」,可以通过以下二维码关注。转载本文请联系卯时卯刻公众号。