亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Java如何獲得泛型類中的泛型類型T.class

 更新時(shí)間:2024年12月14日 15:37:05   作者:nvd11  
獲取泛型符號E的Class對象的方法,1.反射方案利用匿名子類實(shí)例化,但該方法有特殊限制,2.反射方案2直接指定泛型類型,但不靈活,3.構(gòu)造方法方案在初始化時(shí)傳入泛型類,但有寫錯(cuò)風(fēng)險(xiǎn)

Java獲得泛型類的泛型類型T.class

假如我們寫了1個(gè)泛型類MyBase

public class MyBase<E> {
    public MyBase(){
        
    }

    public Class<?> getEClass() {
		?
    }   
}

我想寫1個(gè)getEClass 的方法, 獲取泛型符號E的Class對象。

直接用E.class E.getClass()都是不合法的, 因?yàn)镋只是個(gè)符號, 沒有被實(shí)例化。

1. 反射方案

各種google 各種Stackoverflow 后, 找到利用反射的1個(gè)方法。

public class MyBase<E> {
    public MyBase(){
        
    }

    public Class<?> getEClass() {
        
        //get the Class object of this own class 
        Class<? extends MyBase> thisClass = this.getClass();

        //get the Type Object of supper class
        Type superClassType = thisClass.getGenericSuperclass();
        ParameterizedType pt = (ParameterizedType)superClassType;

        //get the Generic Type array
        Type[] genTypeArr = pt.getActualTypeArguments();
        Type genType = genTypeArr[0];
        if (false == genType instanceof Class){
            return Object.class;
        }

        return (Class<Object>) genType;
    }
}

這個(gè)方法被測試過, 但是有個(gè)很特別的限制,

一定要實(shí)例化匿名子類才有效果。

例如:

   @Test
    void testMyBase() {
        MyBase<String> mybase = new MyBase<String>(){};
        log.info("E class is {}",mybase.getEClass());
    }

注意new MyBase(){} 后面的大括號, 意思不是實(shí)例化MyBase對象, 而是實(shí)例化MyBase的匿名子類。

如果我們寫多1個(gè)具體的子類MyList

public class MyList<E> extends MyBase<E> {
    public MyList(){

	}
}

測試方法:

   @Test
    void testGetEClass() {
        MyList<String> mylist = new MyList<String>();
        log.info("E class is {}",mylist.getEClass());
    }

同樣不行, 原因是

		//get the Type Object of supper class
        Type superClassType = thisClass.getGenericSuperclass();
        ParameterizedType pt = (ParameterizedType)superClassType;

這里 我們只能通過本類的Class對象去獲得1個(gè)父類 ParameterizedType (包括泛型參數(shù)列表的Type)

而JDK本身只提供了getGenericSuperclass()方法去獲得父類的Type 對象

但是沒有提供例如getGenericClass() 去獲取本類的Type對象…

所以一旦我們不實(shí)用匿名子類

當(dāng)下面代碼被執(zhí)行時(shí)

MyList<String> mylist = new MyList<String>();
log.info("E class is {}",mylist.getEClass());

thisClass.getGenericSuperclass() 會(huì)獲得MyList的父類MyBase的Type, 問題是MyBase沒有被實(shí)例化, 所以pt.getActualTypeArguments()[0] 只會(huì)返回E

本身 就return null了

但是如果我們利用匿名子類的化

當(dāng)下面代碼被執(zhí)行時(shí)

MyList<String> mylist = new MyList<String>(){};
log.info("E class is {}",mylist.getEClass());

mylist 的類實(shí)際上是1個(gè)匿名子類MyList$1, 它的父類就是MyList, 而匿名子類的父類在JVM實(shí)際上是被實(shí)例化的

所以pt.getActualTypeArguments()[0] 會(huì)返回泛型具體類String 的Class。

2. 反射方案2

匿名子類畢竟有點(diǎn)奇怪。

如果避免使用你匿名子類, 我們還有1個(gè)辦法

public class MySet extends MyBase<String> {
    public MySet(){
  
    }
}

如上面這個(gè)MySet類, 它直接繼承MyBase時(shí)指定了具體泛型, 令到自己本身不是1個(gè)泛型類。

這樣的話, 不用匿名子類也一樣可以獲得泛型具體類型。

因?yàn)檫@種情況下, 一旦實(shí)現(xiàn)MySet, 因?yàn)镸ySet指定了繼承泛型的具體類型, 在JVM 中MyBase也會(huì)被實(shí)例化。

但是這個(gè)方案一樣不好, 不靈活是其次, 我都會(huì)指定了具體泛型類型了, 何必用再用getEClass()去獲得泛型具體類型? 多次一舉

  @Test
    void testGetEClass2() {
        MySet myset = new MySet();
        log.info("Set E class is {}",myset.getEClass());
    }

3. 構(gòu)造方法方案

這也是外網(wǎng)強(qiáng)烈推薦的方案

public class MyMap<E> {

    private Class<E> eClass;

    public MyMap(Class<E> eClass){
        this.eClass = eClass;
    }

    public Class<E> getEClass(){
        return this.eClass;
    }

}

讓開發(fā)人員在初始化時(shí),順便外部傳入泛型的類, 不過手抖寫錯(cuò)的風(fēng)險(xiǎn)…

測試方法:

  @Test
    void testGetEClass3() {
        MyMap<String> myMap= new MyMap<>(String.class);
        log.info("Set E class is {}",myMap.getEClass());
    }
}

總結(jié)

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。

相關(guān)文章

最新評論