1. 程式人生 > >小師妹學JVM之:JIT中的PrintAssembly續集

小師妹學JVM之:JIT中的PrintAssembly續集

[toc] # 簡介 上篇文章和小師妹一起介紹了PrintAssembly和PrintAssembly在命令列的使用,今天本文將會更進一步講解如何在JDK8和JDK14中分別使用PrintAssembly,並在實際的例子中對其進行進一步的深入理解。 # JDK8和JDK14中的PrintAssembly 小師妹:F師兄,上次你介紹的PrintAssembly的自測命令,怎麼在JDK14中不好用呢? ~~~java java -XX:+UnlockDiagnosticVMOptions -XX:+PrintAssembly -version ~~~ 有什麼不好用的,命令不是正常打出來了嗎? 小師妹:F師兄,你看下我執行的結果,機器碼下面展示的怎麼是448b 5608這樣的數字呀,不應該是assembly language嗎? ![](https://img-blog.csdnimg.cn/20200530194133803.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_0,text_aHR0cDovL3d3dy5mbHlkZWFuLmNvbQ==,size_35,color_8F8F8F,t_70) 嗯...小師妹的話讓我陷入了深深的思考,究竟是什麼導致了這樣的反常的結果呢?是道德的淪喪還是人性的扭曲? 於是我翻遍了baidu,哦,不對是google,還是沒有找到結果。 難點是JDK14有bug?還是JDK14已經使用了另外的Assembly的實現? 有問題就解決問題,我們先從JDK8開始,來探索一下最原始的PrintAssembly的使用。 更多精彩內容且看: * [區塊鏈從入門到放棄系列教程-涵蓋密碼學,超級賬本,以太坊,Libra,比特幣等持續更新](http://www.flydean.com/blockchain/) * [Spring Boot 2.X系列教程:七天從無到有掌握Spring Boot-持續更新](http://www.flydean.com/learn-spring-boot/) * [Spring 5.X系列教程:滿足你對Spring5的一切想象-持續更新](http://www.flydean.com/spring5/) * [java程式設計師從小工到專家成神之路(2020版)-持續更新中,附詳細文章教程](http://www.flydean.com/java-roadmap-2020/) # JDK8中使用Assembly 在JDK8中如果我們執行Assembly的測試命令,可以得到下面的結果: ~~~java java -XX:+UnlockDiagnosticVMOptions -XX:+PrintAssembly -version Java 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) ~~~ 這個故事告訴我們,雖然PrintAssembly開關打開了,但是系統不支援,缺少了hsdis-amd64.dylib檔案。 這個hsdis是一個反彙編的工具,我們需要hsdis的支援才能在JDK8中使用Assembly。 我是mac系統,下面是在mac系統怎麼安裝hsdis: ~~~java hg clone http://hg.openjdk.java.net/jdk8u/jdk8u cd jdk8u/hotspot/src/share/tools/hsdis/ wget http://ftp.heanet.ie/mirrors/ftp.gnu.org/gnu/binutils/binutils-2.30.tar.gz tar -xzf binutils-2.30.tar.gz make 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系統,請自行探索hsdis的安裝方法。 按照步驟先把java8的hsdis-amd64.dylib安裝好。 然後再次執行測試命令: ![](https://img-blog.csdnimg.cn/20200602225754419.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_0,text_aHR0cDovL3d3dy5mbHlkZWFuLmNvbQ==,size_35,color_8F8F8F,t_70) 完美,組合語言出現了。 # JDK14中的Assembly 然後我想到,如果把這個dylib檔案拷貝到JDK14相應的目錄下面,執行一次會怎麼樣呢? > 大家注意,JDK9之後,使用了模組化,所以之前的目錄結構發生了比較大的變化,大家參考上面我列出的地址。 再次執行測試程式碼: ![](https://img-blog.csdnimg.cn/20200602230145355.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_0,text_aHR0cDovL3d3dy5mbHlkZWFuLmNvbQ==,size_35,color_8F8F8F,t_70) 大家看到,Assembly又出現了,真的是讓我熱內盈虧。 其實最開始的時候,我發現JDK14中Assembly沒能正常顯示的時候,我也有想過拷貝一個hsdis-amd64.dylib過來試試,但是一看還需要下載JDK的程式碼,重新編譯,就打起了退堂鼓。 吃一塹,長一智,下次遇到問題千萬不能走捷徑。抄近路害死人呀! # 在JMH中使用Assembly Assembly主要是為了進行程式碼調優或者理解JVM的執行原理來使用的。 這裡我們舉一個在JMH中使用Assembly的例子: ~~~java @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,方便我們檢視和分析結果。 # 總結 本文介紹了JDK8和JDK14中,怎麼開啟PrintAssembly。並舉了一個在JMH中使用的例子。 那麼有人會問了,在JMH中使用Assembly到底有什麼意義呢?別急,我們在後面深入JVM的本質中,馬上就要講到,敬請期待。 > 本文作者:flydean程式那些事 > > 本文連結:[http://www.flydean.com/jvm-jit-printassembly-2/](http://www.flydean.com/jvm-jit-printassembly-2/) > > 本文來源:flydean的部落格 > > 歡迎關注我的公眾號:程式那些事,更多精彩等著您!