隨筆 - 6  文章 - 129  trackbacks - 0
          <2025年6月>
          25262728293031
          1234567
          891011121314
          15161718192021
          22232425262728
          293012345

          常用鏈接

          留言簿(14)

          隨筆檔案(6)

          文章分類(467)

          文章檔案(423)

          相冊

          收藏夾(18)

          JAVA

          搜索

          •  

          積分與排名

          • 積分 - 827218
          • 排名 - 49

          最新評論

          閱讀排行榜

          評論排行榜

          2005-07-12 09:55 作者: robbin 出處: http://dev.yesky.com/241/2033241.shtml

          Hibernate3支持DetachedCriteria,這是一個非常有意義的特性!我們知道,在常規(guī)的Web編程中,有大量的動態(tài)條件查詢,即用戶在網(wǎng)頁上面自由選擇某些條件,程序根據(jù)用戶的選擇條件,動態(tài)生成SQL語句,進行查詢。

            針對這種需求,對于分層應(yīng)用程序來說,Web層需要傳遞一個查詢的條件列表給業(yè)務(wù)層對象,業(yè)務(wù)層對象獲得這個條件列表之后,然后依次取出條件,構(gòu)造查詢語句。這里的一個難點是條件列表用什么來構(gòu)造?傳統(tǒng)上使用Map,但是這種方式缺陷很大,Map可以傳遞的信息非常有限,只能傳遞name和value,無法傳遞究竟要做怎樣的條件運算,究竟是大于,小于,like,還是其它的什么,業(yè)務(wù)層對象必須確切掌握每條entry的隱含條件。因此一旦隱含條件改變,業(yè)務(wù)層對象的查詢構(gòu)造算法必須相應(yīng)修改,但是這種查詢條件的改變是隱式約定的,而不是程序代碼約束的,因此非常容易出錯。

            DetachedCriteria可以解決這個問題,即在web層,程序員使用DetachedCriteria來構(gòu)造查詢條件,然后將這個DetachedCriteria作為方法調(diào)用參數(shù)傳遞給業(yè)務(wù)層對象。而業(yè)務(wù)層對象獲得DetachedCriteria之后,可以在session范圍內(nèi)直接構(gòu)造Criteria,進行查詢。就此,查詢語句的構(gòu)造完全被搬離到web層實現(xiàn),而業(yè)務(wù)層則只負責(zé)完成持久化和查詢的封裝即可,與查詢條件構(gòu)造完全解耦,非常完美!這恐怕也是以前很多企圖在web層代碼中構(gòu)造HQL語句的人想實現(xiàn)的夢想吧!

            示例代碼片段如下:

            web層程序構(gòu)造查詢條件:

            java代碼:

          DetachedCriteria detachedCriteria = DetachedCriteria.forClass(Department.class);
          detachedCriteria.add(Restrictions.eq("name", "department")).createAlias("employees", "e").add(Restrictions.gt(("e.age"), new Integer(20)));

            Department和Employee是一對多關(guān)聯(lián),查詢條件為:

            名稱是“department”開發(fā)部門;
            部門里面的雇員年齡大于20歲;

            業(yè)務(wù)層對象使用該條件執(zhí)行查詢:

            java代碼:

          detachedCriteria.getExecutableCriteria(session).list();

            最大的意義在于,業(yè)務(wù)層代碼是固定不變的,所有查詢條件的構(gòu)造都在web層完成,業(yè)務(wù)層只負責(zé)在session內(nèi)執(zhí)行之。這樣代碼就可放之四海而皆準,都無須修改了。

            然而Spring和Hibernate3的DetachedCriteria有不兼容的問題,因此在Spring環(huán)境下面使用Hibernate3需要注意:

            Spring的HibernateTemplate提供了Hibernate的完美封裝,即通過匿名類實現(xiàn)回調(diào),來保證Session的自動資源管理和事務(wù)的管理。其中核心方法是:

            java代碼:

          HibernateTemplate.execute(new HibernateCallback() {
           public Object doInHibernate(Session session) throws HibernateException {
            ....
           }
          }

            回調(diào)方法提供了session作為參數(shù),有了session,就可以自由的使用Hibernate API編程了。使用了spring的之后,代碼修改如下:

            web層代碼:

            java代碼:

          DetachedCriteria detachedCriteria = DetachedCriteria.forClass(Department.class);
          detachedCriteria.createAlias("employees", "e").add(Restrictions.eq("name", "department")).add(Restrictions.gt(("e.age"), new Integer(20)));
          departmentManager.findByCriteria(detachedCriteria);

            構(gòu)造detachedCriteria,作為參數(shù)傳遞給departmentManager

            業(yè)務(wù)層代碼使用spring,DepartmentManager的findByCriteria如下:

            java代碼:

          public List findByCriteria(final DetachedCriteria detachedCriteria) {
           return (List) getHibernateTemplate().execute(new HibernateCallback() {
            public Object doInHibernate(Session session) throws HibernateException {
             Criteria criteria = detachedCriteria.getExecutableCriteria(session);
             return criteria.list();
            }
           });
          }

            實際上也就是:

            java代碼:

          Criteria criteria = detachedCriteria.getExecutableCriteria(session);
          return criteria.list();

            而已

            但是該程序代碼執(zhí)行,會拋出強制類型轉(zhuǎn)換異常!

            我跟蹤了一下spring和Hibernate源代碼,原因如下:

            spring的HibernateTemplate的execute方法提供的回調(diào)接口具有Session作為參數(shù),但是實際上,默認情況下,HibernateTemplate傳遞給回調(diào)接口的session并不是org.hibernate.impl.SessionImpl類,而是SessionImpl類的一個Proxy類。之所以替換成為一個Proxy類,HibernateTemplate的注釋說明,Proxy提供了一些額外的功能,包括自動設(shè)置Cachable,Transaction的超時時間,Session資源的更積極的關(guān)閉等等。

            java代碼:

          private boolean exposeNativeSession = false;
          ...

            execute方法內(nèi)部:

          Session sessionToExpose = (exposeNativeSession ? session : createSessionProxy(session));

            但是遺憾的是,Hibernate的DetachedCriteria的setExecutableCriteria方法卻要求將session參數(shù)強制轉(zhuǎn)為SessionImpl,但是spring傳過來的卻是一個Proxy類,因此就報錯了。

            java代碼:

          public Criteria getExecutableCriteria(Session session) {
           impl.setSession( (SessionImpl) session ); // 要求SessionImpl,Spring傳遞的是Proxy
           return impl;
          }

            解決方法,禁止Spring的HibernateTemplate傳遞Proxy類,強制要求它傳遞真實的SessionImpl類,即給exexute方法增加一個參數(shù),提供參數(shù)為true,如下:

            java代碼:

          public List findByCriteria(final DetachedCriteria detachedCriteria) {
           return (List) getHibernateTemplate().execute(new HibernateCallback() {
            public Object doInHibernate(Session session) throws HibernateException {
             Criteria criteria = detachedCriteria.getExecutableCriteria(session);
             return criteria.list();
            }
           }, true);
          }


          posted on 2007-12-05 11:10 Ke 閱讀(510) 評論(1)  編輯  收藏 所屬分類: hibernate

          FeedBack:
          # re: Hibernate3的DetachedCriteria支持(轉(zhuǎn))[未登錄] 2008-02-16 10:57 javaboy
          我用spring 中的 DetachedCriteria對象執(zhí)行下面的語句
          DetachedCriteria detachedCriteria = DetachedCriteria.forClass(Department.class);
          detachedCriteria.add(Restrictions.eq("name", "department")).createAlias("employees", "e").add(Restrictions.gt(("e.age"), new Integer(20)));

          返回的List 是一個對象數(shù)組.一個對象元素包含 department 和employee 兩個對象. 請教有沒有方法只返回department 通過 department 延遲加載employee 對象?   回復(fù)  更多評論
            
          主站蜘蛛池模板: 谢通门县| 咸丰县| 车致| 祁连县| 鸡西市| 忻州市| 尤溪县| 镇远县| 定西市| 麦盖提县| 凭祥市| 年辖:市辖区| 迁西县| 成武县| 青岛市| 武冈市| 招远市| 都昌县| 肇源县| 霍林郭勒市| 秭归县| 海盐县| 武乡县| 民丰县| 阳高县| 西吉县| 历史| 长沙市| 宁安市| 德昌县| 牙克石市| 博白县| 宁波市| 米林县| 阿鲁科尔沁旗| 塘沽区| 惠水县| 柯坪县| 乳源| 本溪市| 海盐县|