Tin's Blog

          You are coming a long way, baby~Thinking, feeling, memory...

            BlogJava :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理 ::
            128 隨筆 :: 0 文章 :: 221 評論 :: 0 Trackbacks

          江南白衣的Blog上一篇:
          Java5泛型的用法,T.class的獲取和為擦拭法站臺
          他參考的這里:
          Generic Data Access Objects

          我們的項目中也用的GenericHibernateDAO,里面使用了一個:

          public ?GenericHibernateDAO( final ?Class < E > ?clazz)?{
          ????????
          this .clazz? = ?clazz;
          ????}

          的構造函數。
          但是看了江南白衣的介紹,的確方便的可以寫成:
          public?GenericHibernateDAO()?{
          ????????
          this.clazz?=?(Class<E>)?((ParameterizedType)?getClass()
          ?????????????????????????????????????????????????????????.getGenericSuperclass()).getActualTypeArguments()[
          0];
          ????}
          這樣,繼承的子DAO就可以不用寫Super(xxx.class)進行構造了。

          其中的:
          (Class<E>)?((ParameterizedType)?getClass().getGenericSuperclass()).getActualTypeArguments()[0];?
          非常神奇,看了faint的一個回復(請參照白衣的Blog):
          package?test;?

          import?java.lang.reflect.ParameterizedType;?
          import?java.lang.reflect.Type;?

          import?junit.framework.TestCase;?

          class?TClass<T>?{?
          }?

          class?GoodClass<T>?extends?TClass<String>?{?
          public?ParameterizedType?getClassT()?{?
          return?(ParameterizedType)?getClass().getGenericSuperclass();?
          }?
          }?

          class?BadClass<T>?extends?TClass<T>?{?
          public?ParameterizedType?getClassT()?{?
          return?(ParameterizedType)?getClass().getGenericSuperclass();?
          }?
          }?

          public?class?GenericsTest?extends?TestCase?{?

          private?void?print(Type[]?targs)?{?
          System.out.print(
          "actual?type?arguments?are:");?
          for?(int?j?=?0;?j?<?targs.length;?j++)?{?
          System.out.print(
          "?instance?of?"?+?targs[j].getClass().getName()?+?":");?
          System.out.println(
          "?("?+?targs[j]?+?")");?
          }?
          }?

          public?void?testGoodClass()?throws?Exception?{?
          ParameterizedType?type?
          =?new?GoodClass<String>().getClassT();?
          Type[]?types?
          =?type.getActualTypeArguments();?
          print(types);?

          assertEquals(TClass.
          class,?type.getRawType());?
          assertEquals(String.
          class,?types[0]);?
          }?

          public?void?testBadClass()?throws?Exception?{?
          ParameterizedType?type?
          =?new?BadClass<String>().getClassT();?
          Type[]?types?
          =?type.getActualTypeArguments();?
          print(types);?

          assertEquals(TClass.
          class,?type.getRawType());?
          assertEquals(String.
          class,?types[0]);?
          }?
          }
          例子中的 BadClass 非常有意思,無法獲取T的實際類型,我試驗了半天也得不到。
          看到也有朋友問這個問題:
          http://forum.java.sun.com/thread.jspa?threadID=684429&messageID=3985573

          納悶,怎么就不行呢。
          翻了翻候捷的這篇文章:http://www.jjhou.com/javatwo-2004-GP-in-jdk15.pdf
          才恍然大悟,原來對于BadClass這種情況就是獲取不了它的Class。

          這是擦拭法的本意。
          實際上BadClass<String>()實例化以后Class里面就不包括T的信息了,對于Class而言T已經被擦拭為Object。而真正的T參數被轉到使用T的方法(或者變量聲明或者其它使用T的地方)里面(如果沒有那就沒有存根,這里指ParameterizedTyp),所以無法反射到T的具體類別,也就無法得到T.class。
          而getGenericSuperclass()是Generic繼承的特例,對于這種情況子類會保存父類的Generic參數類型,返回一個ParameterizedType,這時可以獲取到父類的T.class了,這也正是子類確定應該繼承什么T的方法。
          我們應該利用這種特性,這對實現模版方法非常有用。
          posted on 2006-05-08 21:37 Tin 閱讀(2576) 評論(3)  編輯  收藏 所屬分類: Other Project

          評論

          # re: 范型擦拭法即范型類無法獲取范型參數信息的原因 2006-05-28 21:25 差沙
          哦,這么說來faint的例子其實使用了一個障眼法。其實GoodClass里面的String跟new GoodClass<String>中的String沒有任何關系。

          getGenericSuperclass取到的也是extend 時候的String。  回復  更多評論
            

          # re: 范型擦拭法即范型類無法獲取范型參數信息的原因 2006-05-29 16:15 Tin
          恩,我是這么理解的。主要就是因為泛型是編譯期的,運行期本應該是不可用的。獲取getGenericSuperclass則是特例,此時由于編譯順序造成不能編譯后馬上擦除T的信息,所以保留了一個ParameterizedType,所以我們就有幸得到它了。:D  回復  更多評論
            

          # re: 范型擦拭法即范型類無法獲取范型參數信息的原因 2006-11-08 00:51 江南白衣
          另外有些可以情況可以反射,比如如果是在屬性中定義的泛型

          class BookManager{

          Book<String> book;
          }

          這個時候用Field field = BookManager.class.getDeclaredField("book");取得Field
          然后 Type gtype = field.getGenericType();
          if (gtype instanceof ParameterizedType)
          ....同上  回復  更多評論
            

          主站蜘蛛池模板: 徐闻县| 澎湖县| 汉川市| 镇坪县| 远安县| 工布江达县| 灵宝市| 甘谷县| 南皮县| 高清| 新巴尔虎左旗| 九寨沟县| 乐亭县| 和硕县| 双桥区| 宣恩县| 陈巴尔虎旗| 盖州市| 呼图壁县| 宜章县| 柯坪县| 遂川县| 榆林市| 瓦房店市| 潼关县| 榆树市| 石城县| 锦屏县| 中西区| 顺昌县| 莆田市| 灌阳县| 达州市| 大安市| 调兵山市| 固安县| 石柱| 彭阳县| 新龙县| 六盘水市| 湘潭县|