今天,下午在和朋友聊天的時候,聊起了反射這個話題。 我們就從下麵這個段簡單的代碼開始吧。 這個代碼輸出什麼,想必大部分的讀者跟我一樣,會很快地知道答案:0 1 2 3 4 5 6 7 8 9。事實也是如此: 朋友這個時候就提出一個問題,你看在代碼裡面有一個aa();你有什麼辦法,實現這個aa函數,讓 ...
今天,下午在和朋友聊天的時候,聊起了反射這個話題。我們就從下麵這個段簡單的代碼開始吧。
這個代碼輸出什麼,想必大部分的讀者跟我一樣,會很快地知道答案:0 1 2 3 4 5 6 7 8 9。事實也是如此:
朋友這個時候就提出一個問題,你看在代碼裡面有一個aa();你有什麼辦法,實現這個aa函數,讓輸出結果不是這個樣子的(當然 異常不算)。朋友繼續說,你看一下Integer這個裡面的實現。
可以知道的是,Integer 會有一個Cache ,我們要做的,就是修改這個Cache裡面的值。在知道原理之後,我們就可以開始動手實現了,具體代碼如下:
1 private static void aa() {
2 try {
3 // 獲取類
4 Class<?> clazz = Class.forName("java.lang.Integer$IntegerCache");
5
6 // 獲取cache成員變數
7 Field field = clazz.getDeclaredField("cache");
8 field.setAccessible(true);
9 Integer[] cache = (Integer[]) field.get(clazz);
10
11 // Rewrite the Integer cache
12 for (int i = 0; i < cache.length; i++) {
13 cache[i] = new Integer(10+i);
14 }
15 } catch (Exception e) {
16 // TODO: handle exception
17 }
補全這部分代碼之後,我們再執行看看結果:
這樣就完成了對輸出結果的改變。果然是很有意思,只要瞭解得夠多,就足以改變自己的認識。
那回到最開始的問題:什麼是反射呢?
JAVA反射機制是在運行狀態中,對於任意一個類,都能夠知道這個類的所有屬性和方法; 對於任意一個對象,都能夠調用它的任意方法和屬性;這種動態獲取信息以及動態調用對象方法的功能稱為java語言的反射機制。 JAVA反射(放射)機制:“程式運行時,允許改變程式結構或變數類型,這種語言稱為動態語言”。 從這個觀點看,Perl,Python,Ruby是動態語言,C++,Java,C#不是動態語言。但是JAVA有著一個非常突出的動態相關機制:Reflection,用在Java身上指的是我們可以於運行時載入、探知、使用編譯期間完全未知的classes。 換句話說,Java程式可以載入一個運行時才得知名稱的class,獲悉其完整構造(但不包括methods定義),並生成其對象實體、或對其fields設值、或喚起其methods。 這裡引述的是百度百科的解釋。就是動態的,在運行中的時候去處理之前已經寫好的代碼。這樣就提供了很多的靈活性。 後來我們又聊到了 ASM,還有cglib(Code Generation Library)。談到這兩者的應用時,最好的例子 便是Spring的core包裡面的一些內容: 等過段時間,把ASM [1] 和cglib 摸清楚再寫文章詳細說一說,今天就先到這裡了。 _______________________________________________________________________________ [1] (http://asm.ow2.org/,ASM 是一個 Java 位元組碼操控框架。它能夠以二進位形式修改已有類或者動態生成類。ASM 可以直接產生二進位 class 文件,也可以在類被載入入 Java 虛擬機之前動態改變類行為。ASM 從類文件中讀入信息後,能夠改變類行為,分析類信息,甚至能夠根據用戶要求生成新類,快得不要不要的)