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

簡介

上篇文章和小師妹一塊兒介紹了PrintAssembly和PrintAssembly在命令行的使用,今天本文將會更進一步講解如何在JDK8和JDK14中分別使用PrintAssembly,並在實際的例子中對其進行進一步的深刻理解。java

JDK8和JDK14中的PrintAssembly

小師妹:F師兄,上次你介紹的PrintAssembly的自測命令,怎麼在JDK14中很差用呢?linux

java -XX:+UnlockDiagnosticVMOptions -XX:+PrintAssembly -version

有什麼很差用的,命令不是正常打出來了嗎?c++

小師妹:F師兄,你看下我運行的結果,機器碼下面展現的怎麼是448b 5608這樣的數字呀,不該該是assembly language嗎?程序員

嗯...小師妹的話讓我陷入了深深的思考,到底是什麼致使了這樣的反常的結果呢?是道德的淪喪仍是人性的扭曲?spring

因而我翻遍了baidu,哦,不對是google,仍是沒有找到結果。macos

難點是JDK14有bug?仍是JDK14已經使用了另外的Assembly的實現?windows

有問題就解決問題,咱們先從JDK8開始,來探索一下最原始的PrintAssembly的使用。jvm

更多精彩內容且看:模塊化

JDK8中使用Assembly

在JDK8中若是咱們運行Assembly的測試命令,能夠獲得下面的結果:spring-boot

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:

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安裝好。

而後再次運行測試命令:

完美,彙編語言出現了。

JDK14中的Assembly

而後我想到,若是把這個dylib文件拷貝到JDK14相應的目錄下面,運行一次會怎麼樣呢?

你們注意,JDK9以後,使用了模塊化,因此以前的目錄結構發生了比較大的變化,你們參考上面我列出的地址。

再次運行測試代碼:

你們看到,Assembly又出現了,真的是讓我熱內盈虧。

其實最開始的時候,我發現JDK14中Assembly沒能正常顯示的時候,我也有想過拷貝一個hsdis-amd64.dylib過來試試,可是一看還須要下載JDK的代碼,從新編譯,就打起了退堂鼓。

吃一塹,長一智,下次遇到問題千萬不能走捷徑。抄近路害死人呀!

在JMH中使用Assembly

Assembly主要是爲了進行代碼調優或者理解JVM的運行原理來使用的。

這裏咱們舉一個在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,方便咱們查看和分析結果。

總結

本文介紹了JDK8和JDK14中,怎麼開啓PrintAssembly。並舉了一個在JMH中使用的例子。

那麼有人會問了,在JMH中使用Assembly到底有什麼意義呢?別急,咱們在後面深刻JVM的本質中,立刻就要講到,敬請期待。

本文做者:flydean程序那些事

本文連接:http://www.flydean.com/jvm-jit-printassembly-2/

本文來源:flydean的博客

歡迎關注個人公衆號:程序那些事,更多精彩等着您!

相關文章
相關標籤/搜索