擔(dān)心被問到-反射,我來給你壯壯膽
本文轉(zhuǎn)載自微信公眾號「狼王編程」,作者狼王。轉(zhuǎn)載本文請聯(lián)系狼王編程公眾號。
這篇讓我們聊聊什么是反射,為啥好多程序員聽到反射就莫名的害怕呢,本文給你壯壯膽,其實沒多少東西的!
在Java中,想要運行時操作對象的方法、屬性,其中有效的手段,就有反射。這也是框架最常使用的手段,因為框架在編譯時并不知曉系統(tǒng)中有哪些類會被使用。
Java中對象的類型信息在運行時由Class對象表示,Class對象是伴隨類加載而實例化產(chǎn)生的,而反射的實現(xiàn)就圍繞著Class對象。
有很多種方式可以獲取類的Class對象,包括:
- Class<Object> c1 = Object.class;
- Class<?> c2 = Class.forName("java.lang.Object");
- Class<?> c3 = new Object().getClass();
反射可以做什么?
1. 實例化對象
對象的實例化,可以通過new關(guān)鍵字直接實例化,也可以通過反射,例如:
- class.newInstance()
- class.getConstructor(Class<?>... parameterTypes).newInstance(Object ... initargs)
為什么需要用反射來進行對象實例化?
有不能使用new關(guān)鍵字直接實例化對象的場景,例如:Spring容器管理的Bean,只能通過類的全限定名加載類,然后反射實例化。
在不想使用new關(guān)鍵字的場景,目的是為了簡化編程,使代碼美觀,可能大家經(jīng)??吹筋愃频挠梅?,例如:
- public static <T> T parseObject(String text, Class<T> clazz) {
- return parseObject(text, clazz, new Feature[0]);
- }
- /**
- * 簡單的復(fù)制出新類型對象
- */
- public static <S, D> D map(S source, Class<D> destinationClass){
- return mapper.map(source, destinationClass);
- }
2. 篩選合適的類
實際開發(fā)中,經(jīng)常會有這種需要,如果類有某某特征,就進行某某操作。在Spring進行掃描的時候,我們會通過過濾器,來精細(xì)化控制bean的生成,包括:
- 根據(jù)isInstance(Object obj)判斷是否實現(xiàn)某某接口或者繼承特殊的父類;
- 根據(jù)isAnnotationPresent(Class annotationClass)判斷是否被注解標(biāo)記。
3. 方法調(diào)用
有些場景不能或者不合適直接調(diào)用方法,例如:我們處理HTTP請求,需要從URI映射到方法調(diào)用,如果我們能夠窮舉所有的URL到對象方法的映射關(guān)系,那么也沒問題,但是無數(shù)的if條件判斷,顯然不是明智的選擇。
通常我們先會獲取資源對象,然后反射調(diào)用對象的方法。
- Method.invoke(Object obj, Object... args)
如何獲取對象的方法對象,也就是Method對象呢?Class類提供了以下實現(xiàn):
- Method[] getMethods();
- Method[] getDeclaredMethods();
- Method getMethod(String name, Class<?>... parameterTypes);
- Method getDeclaredMethod(String name, Class<?>... parameterTypes);
這些方法可以分為兩類:
- 方法簽名中帶有Declared的,會在當(dāng)前類的所有方法中查找,但不會遍歷父類。
- 不帶Declared的會遍歷所有父類,但只會查找public方法。
推薦使用工具類:org.apache.commons.lang3.reflect.MethodUtils 此類中包含遍歷所有父類查找方法、當(dāng)前類查找public方法或者反射執(zhí)行方法的便捷操作。
4. 屬性操作
- Class.getFields(), Class.getField(String),
- Class.getDeclaredFields(), Class.getDeclaredField(String)
命名規(guī)則同方法,推薦使用工具類:org.apache.commons.lang3.FieldUtils,進行讀取或者賦值操作。
反射進行賦值有幾點需要注意:
- 如果不為public類型,那么設(shè)置字段前必須通過Field.setAccessible(true)方法進行訪問權(quán)限設(shè)置,不然會拋出異常:IllegalAccessException 。
- 如果字段為static類型,那么通過set方法進行賦值時,會忽略obj對象,因為靜態(tài)字段屬于類。
- 如果字段為final類型,不管是public還是 private,那么用set方法賦值時,只有setAccessible進行訪問權(quán)限設(shè)置后,才能正確調(diào)用,不然會報異常:IllegalAccessException。但是對final字段進行set方法賦值時,盡管方法正常調(diào)用,但是并不會改變fianl字段的值。
- 如果字段為final static 類型,那么進行set方法賦值時,總會拋出IllegalAccessException異常。
最后如果對當(dāng)前對象的所有方法,或者所有字段進行某種操作,那么推薦工具類:org.springframework.util.ReflectionUtils
4、總結(jié)
所以反射其實沒那么可怕,消滅恐懼的最好辦法就是面對恐懼,加油學(xué)起來吧!