Tin's Blog

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

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

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

          我們的項(xiàng)目中也用的GenericHibernateDAO,里面使用了一個:

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

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

          其中的:
          (Class<E>)?((ParameterizedType)?getClass().getGenericSuperclass()).getActualTypeArguments()[0];?
          非常神奇,看了faint的一個回復(fù)(請參照白衣的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的實(shí)際類型,我試驗(yàn)了半天也得不到。
          看到也有朋友問這個問題:
          http://forum.java.sun.com/thread.jspa?threadID=684429&messageID=3985573

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

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

          評論

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

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

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

          # re: 范型擦拭法即范型類無法獲取范型參數(shù)信息的原因 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)
          ....同上  回復(fù)  更多評論
            

          主站蜘蛛池模板: 马边| 宁强县| 延吉市| 朝阳县| 德令哈市| 嵩明县| 金阳县| 宣汉县| 平利县| 南宁市| 霍山县| 澜沧| 迭部县| 公安县| 平原县| 西吉县| 铜陵市| 富蕴县| 清镇市| 乌兰浩特市| SHOW| 嵊泗县| 名山县| 于田县| 岳西县| 偃师市| 元谋县| 湘西| 双江| 麦盖提县| 南昌市| 上林县| 隆安县| 兴义市| 盐边县| 丰顺县| 台前县| 临猗县| 额敏县| 常德市| 新疆|