亚洲精品久久久中文字幕-亚洲精品久久片久久-亚洲精品久久青草-亚洲精品久久婷婷爱久久婷婷-亚洲精品久久午夜香蕉

您的位置:首頁技術文章
文章詳情頁

深入分析JAVA 反射和泛型

瀏覽:35日期:2022-08-30 16:33:28

從 JDK5 以后,Java 的 Class 類增加了泛型功能,從而允許使用泛型來限制 Class 類,例如,String.class 的類型實際上是 Class<String>。如果 Class 對應的類暫時未知,則使用 Class<?>。通過在反射中使用泛型,可以避兔使用反射生成的對象需要強制類型轉換。

泛型和 Class 類

使用 Class<T> 泛型可以避免強制類型轉換。例如,下面提供一個簡單的對象工廠,該對象工廠可以根據指定類來提供該類的實例。

public class CrazyitObjectFactory { public static Object getInstance(String clsName) { try { // 創建指定類對應的Class對象 Class cls = Class.forName(clsName); // 返回使用該Class對象所創建的實例 return cls.newInstance(); } catch (Exception e) { e.printStackTrace(); return null; } }}

上面程序中兩行粗體字代碼根據指定的字符串類型創建了一個新對象,但這個對象的類型是 Object,因此當需要使用 CrazyitObjectFactory 的 getInstance() 方法來創建對象時,將會看到如下代碼:

// 獲取實例后需要強制類型轉換Date d = (Date)Crazyit.getInstance('java.util.Date');

甚至出現如下代碼:

JFrame f = (JFrame)Crazyit.getInstance('java.util.Date');

上面代碼在編譯時不會有任何問題,但運行時將拋出 ClassCastException 異常,因為程序試圖將一個 Date 對象轉換成 JFrame 對象。

如果將上面的 CrazyitObjectFactory 工廠類改寫成使用泛型后的 Class,就可以避免這種情況。

public class CrazyitObjectFactory2 { public static <T> T getInstance(Class<T> cls) { try { return cls.newInstance(); } catch (Exception e) { e.printStackTrace(); return null; } } public static void main(String[] args) { // 獲取實例后無須類型轉換 Date d = CrazyitObjectFactory2.getInstance(Date.class); JFrame f = CrazyitObjectFactory2.getInstance(JFrame.class); }}

在上面程序的 getInstance() 方法中傳入一個 Class<T> 參數,這是一個泛型化的 Class 對象,調用該 Class 對象的 newInstance() 方法將返回一個 T 對象,如程序中粗體字代碼所示。接下來當使用 CrazyitObjectFactory2 工廠類的 getInstance() 方法來產生對象時,無須使用強制類型轉換,系統會執行更嚴格的檢查,不會出現 ClassCastException 運行時異常。

前面介紹使用 Array 類來創建數組時,曾經看到如下代碼:

// 使用 Array 的 newInstance 方法來創建一個數組Object arr = Array.newInstance(String.class, 10);

對于上面的代碼其實使用并不是非常方便,因為 newInstance() 方法返回的確實是一個 String[] 數組,而不是簡單的 Object 對象。如果需要將對象當成 String[] 數組使用,則必須使用強制類型轉換——這是不安全的操作。

為了示范泛型的優勢,可以對 Array 的 newInstance() 方法進行包裝。

public class CrazyitArray { // 對Array的newInstance方法進行包裝 @SuppressWarnings('unchecked') public static <T> T[] newInstance(Class<T> componentType, int length) { return (T[]) Array.newInstance(componentType, length); // ① } public static void main(String[] args) { // 使用CrazyitArray的newInstance()創建一維數組 String[] arr = CrazyitArray.newInstance(String.class, 10); // 使用CrazyitArray的newInstance()創建二維數組 // 在這種情況下,只要設置數組元素的類型是int[]即可。 int[][] intArr = CrazyitArray.newInstance(int[].class, 5); arr[5] = '瘋狂Java講義'; // intArr是二維數組,初始化該數組的第二個數組元素 // 二維數組的元素必須是一維數組 intArr[1] = new int[] { 23, 12 }; System.out.println(arr[5]); System.out.println(intArr[1][1]); }}

上面程序中粗體字代碼定義的 newInstance() 方法對 Array 類提供的 newInstance() 方法進行了包裝,將方法簽名改成了 public static <T> T[] newInstance(Class<T> componentType, int length),這就保證程序通過該 newInstance() 方法創建數組時的返回值就是數組對象,而不是 Object 對象,從而避免了強制類型轉換。

提示:程序在①行代碼處將會有一個 unchecked 編譯警告,所以程序使用了 @SuppressWarnings 來抑制這個警告信息。

使用反射來獲取泛型信息

通過指定類對應的 Class 對象,可以獲得該類里包含的所有成員變量,不管該成員變量是使用 private 修飾,還是使用 public 修飾。獲得了成員變量對應的 Field 對象后,就可以很容易地獲得該成員變量的數據類型,即使用如下代碼即可獲得指定成員變量的類型。

// 獲取成員變量 f 的類型Class<?> a = f.getType();

但這種方式只對普通類型的成員變量有效。如果該成員變量的類型是有泛型類型的類型,如 Map<String, Integer> 類型,則不能誰確地得到該成員變量的泛型參數。

為了獲得指定成員變量的泛型類型,應先使用如下方法來獲取該成員變量的泛型類型。

// 獲得成員變量 f 的泛型類型Type gType = f.getGenericType();

然后將 Type 對象強制類型轉換為 ParameterizedType 對象,ParameterizedType 代表被參數化的類型,也就是增加了泛型限制的類型。ParameterizedType 類提供了如下兩個方法。

getRawType():返回沒有泛型信息的原始類型。 getActualTypeArguments():返回泛型參數的類型。

下面是一個獲取泛型類型的完整程序。

public class GenericTest { private Map<String, Integer> score; public static void main(String[] args) throws Exception { Class<GenericTest> clazz = GenericTest.class; Field f = clazz.getDeclaredField('score'); // 直接使用getType()取出的類型只對普通類型的成員變量有效 Class<?> a = f.getType(); // 下面將看到僅輸出java.util.Map System.out.println('score的類型是:' + a); // 獲得成員變量f的泛型類型 Type gType = f.getGenericType(); // 如果gType類型是ParameterizedType對象 if (gType instanceof ParameterizedType) { // 強制類型轉換 ParameterizedType pType = (ParameterizedType) gType; // 獲取原始類型 Type rType = pType.getRawType(); System.out.println('原始類型是:' + rType); // 取得泛型類型的泛型參數 Type[] tArgs = pType.getActualTypeArguments(); System.out.println('泛型信息是:'); for (int i = 0; i < tArgs.length; i++) {System.out.println('第' + i + '個泛型類型是:' + tArgs[i]); } } else { System.out.println('獲取泛型類型出錯!'); } }}

上面程序中的粗體字代碼就是取得泛型類型的關鍵代碼。運行上面程序,將看到如下運行結果:

score的類型是:interface java.util.Map原始類型是:interface java.util.Map泛型信息是:第0個泛型類型是:class java.lang.String第1個泛型類型是:class java.lang.Integer

從上面的運行結果可以看出,使用 getType() 方法只能獲取普通類型的成員變量的數據類型:對于增加了泛型的成員變量,應該使用 getGenericType() 方法來取得其類型。

提示:Type 也是 java.lang.reflect 包下的一個接口,該接口代表所有類型的公共高級接口,Class 是 Type 接口的實現類。Type 包括原始類型、參數化類型、數組類型、類型變量和基本類型等。

以上就是深入分析JAVA 反射和泛型的詳細內容,更多關于JAVA 反射和泛型的資料請關注好吧啦網其它相關文章!

標簽: Java
相關文章:
主站蜘蛛池模板: 亚洲日韩精品欧美一区二区一 | 最新国产v亚洲v欧美v专区 | 成人美女免费网站视频 | 久久美女精品国产精品亚洲 | 国产亚洲精品成人久久网站 | 久久久久久九九 | 久久久久国产 | 成人免费在线 | 伊人久久免费视频 | 国产日韩精品一区二区 | 亚洲国产精品综合久久 | 国产第一页久久亚洲欧美国产 | 国产99免费视频 | 日韩在线观看不卡 | 欧美一级精品高清在线观看 | 欧美日韩亚洲综合在线一区二区 | 久爱www成人网免费视频 | 一区二区视频在线播放 | 免费人成年短视频在线观看免费网站 | 国产视频 一区二区 | 欧美日韩亚洲另类人人澡 | 国产一区二区三区手机在线观看 | japanese日本护士xx亚洲 | 深夜国产| 亚洲小视频网站 | 伊人网综合在线 | 优女视频免费观看 | 91麻豆麻豆| 野战好大好紧好爽视频 | 3至13呦女毛片 | 亚洲成a人片毛片在线 | 在线观看亚洲精品专区 | 久久一区二区三区精品 | 国产亚洲精品一区二区三区 | 成人夜视频 | 国产色婷婷精品综合在线 | 免费黄视频在线观看 | 男人你懂的网站 | 欧美特黄一级高清免费的香蕉 | 毛片黄在线看免费 | 黄色综合 |