本文內容:
- 產生反射技術的需求
- 反射技術的使用
- 一個小示例
首發日期:2018-05-10
產生反射技術的需求:
- 項目完成以后,發現需要增加功能,並且希望增加功能並不需要停止項目運行。
- 在希望不關停項目運行的情況下,於是考慮到將功能都放到一個單獨的項目之外的模塊中,每一個功能實現都從這個模塊中獲取【實際上這個考慮應該是項目開始前就考慮,這個例子可能不是很好】。於是就有了反射的產生。(這種思想有點類似工廠模式,如果學過設計模式的話就明白為什么要做的”開放“)
一個幫助理解例子:
一個網站有計算器功能,剛開始他們只有加減乘除的計算,后來他們決定增加上求乘方、求根號的計算,他們通過“關停網站后修改源代碼“來增加上他們需要的功能。第二次,他們加上xx功能,他們還是關停網站加功能。。。。后來他們發現應該把這個”功能模塊“獨立出來,任何功能實現都通過讀取這個文件來實現,比如想要實現乘方功能就需要查看這個模塊中是否定義了乘方功能。另外,將它獨立出來后,它一個模塊並不影響整個板塊的功能。
【上面的例子中,其他板塊如果想要知道獨立的模塊的內容(屬性、方法),那么就要利用到反射技術,反射技術可以使程序在運行時解析外部對象的信息。】
PS:
- 上面的例子有點像某系統從數據庫中讀取資料,把資料固定到代碼中顯然是一個很不明智的行為,把數據存到數據庫中,再利用SQL語句(類比反射技術)從數據庫中讀取數據就降低了代碼之間的耦合。
- 如果你希望將這個模塊有一個名詞解釋的話,可以把它稱為”配置文件“,有什么功能都可以讀取配置文件來獲取。下面就是講怎么從外部的”配置文件“中讀取具有什么屬性和功能了。
如果你已經了解了 為什么需要反射,那么你可以繼續向下了,如果不了解,那么你可能需要了解更多再深入。
反射的使用:
- 反射技術是一種動態語言的技術,它可以動態的獲取類以及類中的成員,並可以調用該類成員。
獲取外部字節碼文件中的類:
獲取外部類的對象(獲取構造方法):
- 獲取字節碼文件中的類對象后,可以使用newInstance()來生成一個對象。
- 直接使用Class對象.newInstance()生成的對象默認是調用對應類的空參構造方法實例化生成的對象。
- 但沒有空參數的構造方法時:
- 1.首先需要獲取構造方法來生成指定的對象,可以使用getConstructors()來獲取所有構造函數,getConstructor(參數類型列表)來獲取符合參數列表的構造方法;【1.這里對於getConstructor注意參數類型列表不是數據類型+變量名,而是一個個Class:數據類型.class】
【2.Constructor是一個存放構造器的類。】【3.獲取的構造方法應該是public的】
- 2.然后再利用構造器對象.newInstance(參數列表)來實例化對象。
- 如果是非public的構造方法,可以使用
getDeclaredConstructor(參數類型列表)來獲取構造方法。同樣的,
getDeclaredConstructors()
是獲取所有不論public還是非public的構造方法。
獲取外部對象的屬性:
- 首先通過Class對象獲取變量域field:
【Field類用來存儲變量域】
- getField(變量名):獲取指定變量名的public的成員變量
- getFields():獲取所有public的成員變量
- getDeclaredField(變量名):獲取不論public還是非public的指定變量名的成員變量
- getDeclaredFields():獲取不論public還是非public的所有成員變量
- 然后使用Field對象來操作:
獲取外部對象的方法:
- 首先通過Class對象獲取方法域method:
【Field類用來存儲方法域】
- getMethod(方法名, 參數類型列表):獲取public的指定名的方法
- getMethods():獲取public的所有方法
getDeclaredMethod(方法名, 參數類型列表):獲取不論public還是非public的指定名的方法【用來獲取私有的方法】
getDeclaredMethods():獲取不論public還是非public的方法
然后使用Method對象操作:
補充:
- 如果想使用普通方法就能訪問私有的變量或屬性,可以使用:field.setAccesccible(true)
一個小示例:
1.讀取”軟件列表“:
2.通過讀取功能列表來獲知類的字節碼的路徑
3.開發"軟件”:
3.給軟件列表加上內容:
4.運行