查看運(yùn)行中Java Class 源碼的三板斧
線上跑了個(gè)應(yīng)用,總感覺(jué)運(yùn)行結(jié)果和你預(yù)期的有出入,而且代碼似乎和你本地的也不一致。這可咋辦,一咬牙,一跺腳停機(jī)重新上線嗎?
那是夠折騰的,我們來(lái)了解幾種不停機(jī)來(lái)查看 Java 應(yīng)用Class 源碼的方式。
像程咬金的功夫一樣,針對(duì)Class 源碼查看,我們也有三板斧。
一般我們要查看一個(gè)運(yùn)行中程序的 Class 源碼,大致會(huì)經(jīng)過(guò)這樣三步:
一、定位,找到這個(gè)Class
二、將遠(yuǎn)程 的 class 文件下載到本地
三、反編譯,查看源碼
這的過(guò)程,應(yīng)該都很清楚,我們重點(diǎn)來(lái)看第一第二步,怎樣找到這個(gè)class并下載到本地。
對(duì)于找Class 來(lái)說(shuō),如果是我們自己開(kāi)發(fā)的代碼,而不是編譯時(shí)或者運(yùn)行時(shí)增強(qiáng)以及生成的,那直接在遠(yuǎn)程服務(wù)器搜索到并下載到本地就OK了。
像那些編譯時(shí)、以及運(yùn)行時(shí)增強(qiáng)過(guò)的Class,甚至動(dòng)態(tài)生成的Class,比如像OpenJPA、Spring/Cglib 的動(dòng)態(tài)代理, 上面的辦法就無(wú)能為力了。
這種情況,可以直接用JDK 的神器 SA 工具(Java虛擬機(jī)的顯微鏡 Serviceability Agent),在 Class browser 里直接搜索,然后選擇 create .class file 文件就會(huì)默認(rèn)保存到 SA Jar 所在的目錄。不過(guò)缺點(diǎn)是,SA連接時(shí)會(huì)把應(yīng)用掛起,直到斷開(kāi)。
還有一種方法是各種增強(qiáng)的類(lèi)庫(kù),都默認(rèn)提供了保存生成類(lèi)的配置,如果此選項(xiàng)打開(kāi),就會(huì)把生成的、增強(qiáng)的class 寫(xiě)到本地。比如像Cglib,在System.property里設(shè)置即可。
System.setProperty(DebuggingClassWriter.DEBUG_LOCATION_PROPERTY, ".");
Cglib 運(yùn)行時(shí)判斷配置,打開(kāi)就會(huì)把生成類(lèi)的 class 二進(jìn)制文件寫(xiě)到本地。
如果類(lèi)庫(kù)沒(méi)有或者你沒(méi)找到這個(gè)選項(xiàng),又確實(shí)想要查看,那可以 Attach 一個(gè) Java Agent 到JVM上。用Instrument或者ByteBuddy免 Jar 包掛上去,參考之前的文章(不用Jar 包的Agent?幾行代碼實(shí)現(xiàn)運(yùn)行時(shí)增強(qiáng))
有了Agent 的 transformer,大顯身手就看你的了。
本文轉(zhuǎn)載自微信公眾號(hào)「Tomcat那些事兒 」,可以通過(guò)以下二維碼關(guān)注。轉(zhuǎn)載本文請(qǐng)聯(lián)系Tomcat那些事兒公眾號(hào)。