在mac上使用
查看java安裝目錄位置
輸入命令 /usr/libexec/java_home
/Library/Java/JavaVirtualMachines/jdk1.8.0_191.jdk/Contents/Home
1 Jps虛擬機(jī)進(jìn)程狀態(tài)工具
jps [options] [hostId] hostid(ip:port)
參數(shù)說明
-q 只輸出LVMID 省略主類的名稱
-m 輸出虛擬機(jī)進(jìn)程啟動時傳遞給主類main()函數(shù)的參數(shù)
-l 輸出主類全名 如果進(jìn)程執(zhí)行的是jar包 輸出jar路徑
-v 輸出虛擬機(jī)進(jìn)程啟動時jvm參數(shù)
2 jit 生成代碼反匯編
使用時遇到
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
異常 需要下載依賴庫。
下載https://github.com/evolvedmicrobe/benchmarks/blob/master/hsdis-amd64.dylib
下載上面的庫文件放到j(luò)ava_home/jre/lib下即可。
反匯編(Disassembly):把目標(biāo)代碼轉(zhuǎn)為匯編代碼的過程,也可以說是把機(jī)器語言轉(zhuǎn)換為匯編語言代碼、低級轉(zhuǎn)高級的意思,常用于軟件破解(例如找到它是如何注冊的,從而解出它的注冊碼或者編寫注冊機(jī))、外掛技術(shù)、病毒分析、逆向工程、軟件漢化等領(lǐng)域。學(xué)習(xí)和理解反匯編語言對軟件調(diào)試、漏洞分析、OS的內(nèi)核原理及理解高級語言代碼都有相當(dāng)大的幫助,在此過程中我們可以領(lǐng)悟到軟件作者的編程思想。總之一句話:軟件一切神秘的運(yùn)行機(jī)制全在反匯編代碼里面。
分析代碼程序執(zhí)行語義(做了什么)在字節(jié)碼層面就行了,分析程序代碼執(zhí)行行為(虛擬機(jī)怎么做的 性能如何)在字節(jié)碼層面就不行了需要HSDIS插件
Java -XX:+PrintAssembly -Xcomp -XX:CompileCommand=dontinline,Bar.sum -XX:CompileCommand=compileonly,Bar.sum test.Bar
-Xcomp 讓虛擬機(jī)以編譯模式執(zhí)行代碼
-XX:CompileCommand=dontinline,*Bar.sum 不內(nèi)聯(lián)sum()
-XX:CompileCommand=compileonly,*Bar.sum 只編譯sum()
-XX:+PrintAssembly 輸出反匯編內(nèi)容
java -XX:+UnlockDiagnosticVMOptions -XX:+PrintAssembly -Xcomp -XX:CompileCommand=dontinline,Bar.sum -XX:CompileCommand=compileonly,Bar.sum
查看是否可以正常使用hsdis
java -XX:+UnlockDiagnosticVMOptions -XX:+PrintAssembly -version
參數(shù)說明
-XX:+UnlockExperimentalVMOptions 和-XX:+UnlockDiagnosticVMOptions ;來解鎖任何額外的隱藏參數(shù)
執(zhí)行命令時候出現(xiàn)異常zsh: no matches found: -XX:CompileCommand=dontinline,*Bar.sum
需要配置了下~/.zshrc
unsetopt nomatch
二、JITWatch安裝:
1.下載源碼:
git clone https://github.com/AdoptOpenJDK/jitwatch.git
2.編譯:
cd jitwatch
ant clean compile test run #已安裝maven環(huán)境的,也可以執(zhí)行mvn clean compile test exec:java
3.啟動:(仍在jitwatch目錄下)
sh launchUI.sh
- mov %eax,-0x8000(%esp):檢查棧溢。
- push %ebp:保存上一棧幀基址。
- sub $0x18,%esp:給新幀分配空間。
- mov 0x8(%ecx),%eax:取實例變量a,這里0x8(%ecx)就是ecx+0x8的意思,前面“[Constants]”節(jié)中提示了“this:ecx = 'test/Bar'”,即ecx寄存器中放的就是this對象的地址。偏移0x8是越過this對象的對象頭,之后就是實例變量a的內(nèi)存位置。這次是訪問“Java堆”中的數(shù)據(jù)。
- mov $0x3d2fad8,%esi:取test.Bar在方法區(qū)的指針。
- mov 0x68(%esi),%esi:取類變量b,這次是訪問“方法區(qū)”中的數(shù)據(jù)。
- add %esi,%eax 、add %edx,%eax:做2次加法,求a+b+c的值,前面的代碼把a(bǔ)放在eax中,把b放在esi中,而c在[Constants]中提示了,“parm0:edx = int”,說明c在edx中。
- add $0x18,%esp:撤銷棧幀。
- pop %ebp:恢復(fù)上一棧幀。
- test %eax,0x2b0100:輪詢方法返回處的SafePoint
- ret:方法返回。