風人園

          弱水三千,只取一瓢,便能解渴;佛法無邊,奉行一法,便能得益。
          隨筆 - 99, 文章 - 181, 評論 - 56, 引用 - 0

          導航

          <2025年7月>
          293012345
          6789101112
          13141516171819
          20212223242526
          272829303112
          3456789

          常用鏈接

          留言簿(11)

          隨筆分類

          隨筆檔案

          文章分類

          文章檔案

          新聞檔案

          收藏夾

          友情鏈接

          搜索

          •  

          最新評論

          閱讀排行榜

          評論排行榜

          SpringSide中使用的JDK5.0特性

          SpringSide中使用的JDK5.0特性

          ??????隨著光陰推移,Annotation 慢慢在開源框架中推廣,泛型漸漸被程序員們用熟,加上AutoBoxing的小糖,SpringSide終于離不開JDK5.0。

          1.AutoBoxing 與 For Each 循環

          ??? 本來int的非Object性就很無聊,在JDK5.0終于提供了autoboxing功能。這個語法簡化糖,被用在了每一個地方。

          ??? for each 循環也改善了原本總要愣一下的collection遍歷。不過對于非JDK基本類型,collection必須用泛型聲明,如List<Book>。

          2. 泛型

          ??? 泛型大量用于SpringSide Core中的基類,使子類更簡潔,基類更強大。當然,基類是難讀了,所以才需要社區花上這么長的時間來把<T>看到順眼。

          ??? 泛型使用的有兩個定式:

          2.1 避免強制類型轉換

          ???如果函數輸入參數里含Class類型,而返回值又是該Class的實體,應該將該函數設為泛型函數。最典型的例子是HibernateGenericDao的get() 函數

          								public <T> T get(Class<T> entityClass, Serializable id) {
             return (T) getHibernateTemplate().get(entityClass, id);
          }

          ? 其中眼花繚亂的第一個<T>聲明這是一個泛型函數,第2個T聲明返回值為T,第三個Class<T>代表 T.class。基類寫的辛苦,但子類用得爽快?

          ???Book book = (Book)manager.get(Book.class,1) 簡化成了?Book book = manager.get(Book.class,1);

          2. 2 泛型配合反射API從T獲得 T.class。

          ?? 最典型的例子HibernateEntityDao,子類只需以下定義,即獲得要管理的Entity的Class。

          BookManager extends HibernateEntityDao<Book>

          ??? 此時子類只要聲明一次T,上面的Book book = (Book)manager(Book.class,1) 就能簡化成Book = manager.get(1);

          ??? 一舉兩得地既避免了強制類型轉換,又聲明了T.class 供框架使用,無須再在Manager的構造函數或getEntityClass()函數定義entityClass,。

          ??? 反射的API 詳見GenricsUtils ,精簡的對上面BookManager的定義反射代碼如下:

          Type genType = clazz.getGenericSuperclass();
          Type [] params = ((ParameterizedType) genType).getActualTypeArguments();
          return (Class) params[0];

          ???泛型反射的關鍵是獲取ParameterizedType,再調用它的getActualTypeArguments()方法獲得實際綁定的類型。但注意public class BookManager<Book>是不能被反射的,因為擦拭法的緣故。只有在Superclass 或者成員變量(Field.getGenericType())等有函數返回ParameterizedType的時候才能成功反射,,

          比如?

          								public class BookManager extends Manager<Book>{}?public class?BookAction {??? private BookManager<Book> manager;}

          ?2.3 其他應用

          ?? 1. 在XFire中,List getBooksByCategory()函數返回的結果,需要用aegis.xml 文件聲明List中的元素為Book.

          ?? 而如果定義函數為 List<Book> getBooksByCategory(),就不再需要聲明,省掉XMl配置文件。

          3.Annotation

          ???? Annotation 大幅提升了Java的編程模式,SpringSide 目前運用的Annotation 有以下三個地方,幸運的是前兩者同時也提供JDK1.4的JAVADoc式配置,只是麻煩一點點而已

          ???? 3.1. Hibernate Annotation

          ???????? 使用Hibernate Annotation 代替hbm文件,因為annotation高度的默認性,典型的POJO基本上不需要定義什么,代碼的簡約性和可管理性大幅提高,直追ROR。

          ???????? 另外,經過測試,annotation 完全能勝任一些比較復雜的Mapping定義,如Product-Book的父子繼承關系,Order-OrderItem-Product的經典三角關系。

          ???? 2. XFire JSR181 Annotation

          ?????????JSR181聲明的Web Service,比原本用xml定義的模式節約了XML文件和配置代碼的數量。

          ???? 3.?聲明Entity類型的Annotation

          ?????????使用Annotation 聲明Entity的類型,比如Udeletable,Auditable 等,比用接口聲明的方式有更少的侵入性,詳見 侵入,非侵入?Interface vs Annotation

          4. 三種內置Annotation

          JDK5.0 有SuppressWarnings,Deprecated和Override 三種內置的annotation:

          ? @Override

          ?? 此標簽一方面提醒用戶這是個重載函數,另一方面保證了父類函數的參數或者名字改變時,子類如果沒有跟著變化,就會編譯不過。

          ?? 雖然有點占地方,但用處的確很大,不會哪天子類被人賣了都不知道。

          ? 所以我設置了讓IDEA6檢查所有重載函數必須加上@Override標識。

          ? @SuppressWarnings("unchecked")

          ??? 此標簽可以讓編譯器忽略某種warning信息,比如減少JDK5.0的集合操作引入范型后無處不在的warning。

          ??? 因為有些非JDK5.0的開源庫如hibernate, 函數返回的一定是List,而不會是List<User>,這時候IDE就會爆出很多warning。用SuppressWarning("unchecked")可以讓IDE安靜一些。

          ??? 其他常用warning還包括 @SuppressWarnings("unused") 和 SuppressWarnings("serial")

          ? @Deprecated

          ?? 此標簽以前寫在JavaDoc里,現在提到annotation,注釋已廢棄的函數。用戶使用該函數的話,編譯時會得到"你用了廢棄"的提示。

          5.可變參數

          用于HibernateGenericDao中,簡化函數接口。

          比如 一個public List find(String hql, Object... values),就支持了如下四種調用,避免了以前的煞費苦心的定義多種接口,然后把參數轉成統一模式的大量重復定義。

          dao.find(hql);
          dao.find(hql,arg0);
          dao.find(hql,arg0,arg1);
          dao.find(hql,new Object[arg0,arg1])
          

          posted on 2006-12-16 15:28 風人園 閱讀(366) 評論(0)  編輯  收藏 所屬分類: Java

          主站蜘蛛池模板: 巴南区| 鹤岗市| 齐齐哈尔市| 自贡市| 鄱阳县| 监利县| 绥宁县| 肥城市| 米易县| 孝昌县| 义马市| 南丰县| 亳州市| 平乡县| 华坪县| 梁山县| 太原市| 会理县| 讷河市| 广河县| 沐川县| 永清县| 汝阳县| 两当县| 鹤山市| 伊通| 五大连池市| 都兰县| 如皋市| 内乡县| 句容市| 贡觉县| 武城县| 叶城县| 土默特左旗| 隆尧县| 榆社县| 南开区| 彰化市| 刚察县| 育儿|