ASM: 一個低調(diào)成功者的自述
我就是那個不太知名的ASM, 你可能聽說過我,也可能完全不知道我。
但是你要是認為我無足輕重,那就大錯特錯了。
沒有我, 你經(jīng)常使用的Spring , hibernate 將會失去最核心的功能。
沒有我, Jython , JRuby 根本就移植不到Java 虛擬機中來。
沒有我, Clojure, Groovy這些時髦的語言也沒法在Java 虛擬機中運行。
一句話來概括我的能力: 我可以動態(tài)的修改已經(jīng)編譯過的class , 還可以動態(tài)的生成新的java class, 注意我說的動態(tài)這個詞, 那可以是完全在運行時, 在內(nèi)存中完成的, 這是一件非常厲害的本事。
既然我是為了動態(tài)的修改class 文件而生, 為什么叫做ASM呢?
其實我的創(chuàng)造者在創(chuàng)造我的時候, 為了起名也是大費周章,后來他突然想到C語言中的__asm__ 這個關(guān)鍵字, 可以允許你們在C語言中寫點匯編, 他就把ASM這個關(guān)鍵字挪用了。
考慮到起名字確實是一件非常折磨人的事情, 我也就忍了, 就叫這個ASM吧, 至少能體現(xiàn)出我位于系統(tǒng)的***層, 不, 應(yīng)該說是最基礎(chǔ)的層次。
我聽到下面有人問了, 不就是動態(tài)生成類嗎? 我完全可以像jsp那樣, 使用JavaComplier接口在運行時動態(tài)的編譯一個java 源代碼, 這不也是動態(tài)生成class嗎?
其實不一樣, 你那是生成新的類, 能對現(xiàn)有的class 進行修改嗎?
又有人發(fā)話了, 你為什么要動態(tài)的在運行時來修改類啊? 我為什么不能在編譯以前就把類的功能都寫好啊?
當(dāng)然,你要是能把所有的功能都寫好,那自然不錯,但是人生之不如意,十之八九啊。
舉一個最最極端的例子,你從別人那里獲得了一個沒有源代碼的jar 文件, 你想對其中的一個class 進行增強,腫么辦? 你可能說: 我可以反編譯啊.. 算了吧, 反編譯的代碼能看嗎?
還有你們經(jīng)常掛在嘴邊的AOP, 在配置文件中聲明一些功能例如事務(wù)支持, 然后要這些功能動態(tài)的織入到業(yè)務(wù)代碼中,腫么辦? 有人說我可以用Java 動態(tài)代理啊, 是, 你可以用,但是人家要是沒有接口不還是得瞪眼干著急?
順便吐個槽,你們AOP的那些術(shù)語實在是太爛, 什么PointCut, 什么Advice, 除了把人搞暈,還有什么用處?
對了, 還有那個Hibernate , 難道你不知道那個所謂的實體類是被我給增強過的? 你使用的并不是你看到的, 懂了么?
那到底是怎么實現(xiàn)的動態(tài)修改類的? 其實很簡單, 去下載一個Java 虛擬機規(guī)范, 花上半年時間, 把每個字都搞懂了, ***像黑客帝國中的Neo那樣, 看到的整個世界都是二進制流, 你自然明白我是怎么做的了。
我給你簡單的說一下: 我的核心呢主要是三個類, ClassReader, ClassWriter , ClassVisitor, 你用這三個家伙就可以去解析一個class 字節(jié)碼, 獲得字段了,方法了等信息, 當(dāng)然最重要的可以對這些信息進行修改, 最終形成一個代表新class 的字節(jié)碼數(shù)組, 剩下的事估計你就知道怎么做了, 其實也不屬于我ASM了。
還不知道? 很簡單嘛, 就是用個ClassLoader 把這個字節(jié)碼數(shù)組Load到虛擬機中, 然后通過反射一調(diào)用,不就完了?
對了, 我的創(chuàng)造者使用了Visitor模式來設(shè)計這個API, 說實在的,設(shè)計的還真不錯。 細節(jié)太多 ,我在這里就不羅嗦了, 感興趣的同學(xué)可以看看 (http://download.forge.objectweb.org/asm/asm4-guide.pdf ) , 絕對物超所值, 當(dāng)然你的英文要好。
說了這么多,還是上一點代碼吧, 讓你對我有個直觀的認識:
我估計你也看不明白, 其實就是hello world了:
你可能心里在想, word哥, 想使用你ASM需要非常透徹的理解Java 虛擬機指令和Java虛擬機內(nèi)部結(jié)構(gòu)才能使用啊。
沒錯, 我剛剛說過, 需要讀懂Java 虛擬機規(guī)范 , 我負責(zé)處理的是最基礎(chǔ)的東西 , 很多碼農(nóng)并不會直接使用我來編程。
有個叫CGLib的家伙在我的基礎(chǔ)之上做了不錯的封裝, 他做了一些高級的API抽象, 讓普通的程序員也能夠比較簡單的對一個現(xiàn)成的類的行為進行修改。 由于更容易使用, CGLib的用戶反而更多, 像Spring , Hibernate 。。。 我的風(fēng)頭完全被蓋過了, 我想這就是我為什么這么低調(diào)、默默無聞的原因。
但是我初心不改 ,堅持對***層的字節(jié)碼進行操作, 我持續(xù)優(yōu)化, 讓自己變的又小又快, 為別的軟件提供最有力的支持。
一開始我就說了Clojure對吧, 這是一個函數(shù)式編程的語言, 是解釋執(zhí)行的,沒有編譯過程, 那他憑什么能運行在Java 虛擬機上? 不就是利用我ASM來動態(tài)的生成字節(jié)碼 嗎? Jython, JRuby, Groovy 也是大同小異。
最近Java 帝國給我頒發(fā)了一個特殊貢獻勛章, 獎勵我對繁榮Java 虛擬機市場做出的重大貢獻, 不謙虛的說, 這絕對是名副其實, 原來JVM中只有一門語言,那就是Java , 你看看現(xiàn)在語言多的都選不多來了。
好, 今天就說到這里吧, 下次再會。
【本文為51CTO專欄作者“劉欣”的原創(chuàng)稿件,轉(zhuǎn)載請通過作者微信公眾號coderising獲取授權(quán)】