小师妹学JVM之:JIT中的PrintAssembly续集
文章目錄
- 簡(jiǎn)介
- JDK8和JDK14中的PrintAssembly
- JDK8中使用Assembly
- JDK14中的Assembly
- 在JMH中使用Assembly
- 總結(jié)
簡(jiǎn)介
上篇文章和小師妹一起介紹了PrintAssembly和PrintAssembly在命令行的使用,今天本文將會(huì)更進(jìn)一步講解如何在JDK8和JDK14中分別使用PrintAssembly,并在實(shí)際的例子中對(duì)其進(jìn)行進(jìn)一步的深入理解。
JDK8和JDK14中的PrintAssembly
小師妹:F師兄,上次你介紹的PrintAssembly的自測(cè)命令,怎么在JDK14中不好用呢?
java -XX:+UnlockDiagnosticVMOptions -XX:+PrintAssembly -version有什么不好用的,命令不是正常打出來(lái)了嗎?
小師妹:F師兄,你看下我運(yùn)行的結(jié)果,機(jī)器碼下面展示的怎么是448b 5608這樣的數(shù)字呀,不應(yīng)該是assembly language嗎?
嗯…小師妹的話讓我陷入了深深的思考,究竟是什么導(dǎo)致了這樣的反常的結(jié)果呢?是道德的淪喪還是人性的扭曲?
于是我翻遍了baidu,哦,不對(duì)是google,還是沒有找到結(jié)果。
難點(diǎn)是JDK14有bug?還是JDK14已經(jīng)使用了另外的Assembly的實(shí)現(xiàn)?
有問題就解決問題,我們先從JDK8開始,來(lái)探索一下最原始的PrintAssembly的使用。
更多精彩內(nèi)容且看:
- 區(qū)塊鏈從入門到放棄系列教程-涵蓋密碼學(xué),超級(jí)賬本,以太坊,Libra,比特幣等持續(xù)更新
- Spring Boot 2.X系列教程:七天從無(wú)到有掌握Spring Boot-持續(xù)更新
- Spring 5.X系列教程:滿足你對(duì)Spring5的一切想象-持續(xù)更新
- java程序員從小工到專家成神之路(2020版)-持續(xù)更新中,附詳細(xì)文章教程
JDK8中使用Assembly
在JDK8中如果我們運(yùn)行Assembly的測(cè)試命令,可以得到下面的結(jié)果:
java -XX:+UnlockDiagnosticVMOptions -XX:+PrintAssembly -versionJava HotSpot(TM) 64-Bit Server VM warning: PrintAssembly is enabled; turning on DebugNonSafepoints to gain additional output Could not load hsdis-amd64.dylib; library not loadable; PrintAssembly is disabled java version "1.8.0_171" Java(TM) SE Runtime Environment (build 1.8.0_171-b11) Java HotSpot(TM) 64-Bit Server VM (build 25.171-b11, mixed mode)這個(gè)故事告訴我們,雖然PrintAssembly開關(guān)打開了,但是系統(tǒng)不支持,缺少了hsdis-amd64.dylib文件。
這個(gè)hsdis是一個(gè)反匯編的工具,我們需要hsdis的支持才能在JDK8中使用Assembly。
我是mac系統(tǒng),下面是在mac系統(tǒng)怎么安裝hsdis:
hg clone http://hg.openjdk.java.net/jdk8u/jdk8ucd jdk8u/hotspot/src/share/tools/hsdis/wget http://ftp.heanet.ie/mirrors/ftp.gnu.org/gnu/binutils/binutils-2.30.tar.gztar -xzf binutils-2.30.tar.gzmake BINUTILS=binutils-2.30 ARCH=amd64#java8 sudo cp build/macosx-amd64/hsdis-amd64.dylib /Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/jre/lib/server/#java9 onwards sudo cp build/macosx-amd64/hsdis-amd64.dylib /Library/Java/JavaVirtualMachines/jdk-9.0.4.jdk/Contents/Home/lib/server/如果你是linux或者windows系統(tǒng),請(qǐng)自行探索hsdis的安裝方法。
按照步驟先把java8的hsdis-amd64.dylib安裝好。
然后再次運(yùn)行測(cè)試命令:
完美,匯編語(yǔ)言出現(xiàn)了。
JDK14中的Assembly
然后我想到,如果把這個(gè)dylib文件拷貝到JDK14相應(yīng)的目錄下面,運(yùn)行一次會(huì)怎么樣呢?
大家注意,JDK9之后,使用了模塊化,所以之前的目錄結(jié)構(gòu)發(fā)生了比較大的變化,大家參考上面我列出的地址。
再次運(yùn)行測(cè)試代碼:
大家看到,Assembly又出現(xiàn)了,真的是讓我熱內(nèi)盈虧。
其實(shí)最開始的時(shí)候,我發(fā)現(xiàn)JDK14中Assembly沒能正常顯示的時(shí)候,我也有想過拷貝一個(gè)hsdis-amd64.dylib過來(lái)試試,但是一看還需要下載JDK的代碼,重新編譯,就打起了退堂鼓。
吃一塹,長(zhǎng)一智,下次遇到問題千萬(wàn)不能走捷徑。抄近路害死人呀!
在JMH中使用Assembly
Assembly主要是為了進(jìn)行代碼調(diào)優(yōu)或者理解JVM的運(yùn)行原理來(lái)使用的。
這里我們舉一個(gè)在JMH中使用Assembly的例子:
@Warmup(iterations = 2, time = 1, timeUnit = TimeUnit.SECONDS) @Measurement(iterations = 2, time = 1, timeUnit = TimeUnit.SECONDS) @Fork(value = 1,jvmArgsPrepend = {"-XX:+UnlockDiagnosticVMOptions","-XX:CompileCommand=print,com.flydean.PrintAssemblyUsage::testPrintAssembly" } ) @State(Scope.Benchmark) @BenchmarkMode(Mode.AverageTime) @OutputTimeUnit(TimeUnit.NANOSECONDS) public class PrintAssemblyUsage {int x;@Benchmark@CompilerControl(CompilerControl.Mode.DONT_INLINE)public void testPrintAssembly() {for (int c = 0; c < 1000; c++) {synchronized (this) {x += 0xFF;}}}public static void main(String[] args) throws RunnerException {Options opt = new OptionsBuilder().include(PrintAssemblyUsage.class.getSimpleName()).build();new Runner(opt).run();} }上面的例子中,我們使用了-XX:CompileCommand指定要打印的方法,而不是輸出所有的Assembly,方便我們查看和分析結(jié)果。
總結(jié)
本文介紹了JDK8和JDK14中,怎么開啟PrintAssembly。并舉了一個(gè)在JMH中使用的例子。
那么有人會(huì)問了,在JMH中使用Assembly到底有什么意義呢?別急,我們?cè)诤竺嫔钊隞VM的本質(zhì)中,馬上就要講到,敬請(qǐng)期待。
本文作者:flydean程序那些事
本文鏈接:http://www.flydean.com/jvm-jit-printassembly-2/
本文來(lái)源:flydean的博客
歡迎關(guān)注我的公眾號(hào):程序那些事,更多精彩等著您!
總結(jié)
以上是生活随笔為你收集整理的小师妹学JVM之:JIT中的PrintAssembly续集的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 区块链系列教程之:比特币中的共识
- 下一篇: 深入理解编译优化之循环展开和粗化锁