<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    隨筆-86  評論-33  文章-0  trackbacks-0
    Hibernate3支持DetachedCriteria,這是一個非常有意義的特性!我們知道,在常規的Web編程中,有大量的動態條件查詢,即用戶在網頁上面自由選擇某些條件,程序根據用戶的選擇條件,動態生成SQL語句,進行查詢。

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

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

      示例代碼片段如下:

      web層程序構造查詢條件:

      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是一對多關聯,查詢條件為:

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

      業務層對象使用該條件執行查詢:

      java代碼:
    detachedCriteria.getExecutableCriteria(session).list();


      最大的意義在于,業務層代碼是固定不變的,所有查詢條件的構造都在web層完成,業務層只負責在session內執行之。這樣代碼就可放之四海而皆準,都無須修改了。

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

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

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


      回調方法提供了session作為參數,有了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);


      構造detachedCriteria,作為參數傳遞給departmentManager

      業務層代碼使用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(); 


      而已

      但是該程序代碼執行,會拋出強制類型轉換異常!

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

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

      java代碼:
    private boolean exposeNativeSession = false


      execute方法內部:
    Session sessionToExpose = (exposeNativeSession ? session : createSessionProxy(session)); 

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

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



      解決方法,禁止Spring的HibernateTemplate傳遞Proxy類,強制要求它傳遞真實的SessionImpl類,即給exexute方法增加一個參數,提供參數為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); 



    //統計記錄數
    Criteria criteria = detachedCriteria.getExecutableCriteria(session); 
    int totalCount = ((Integer) criteria.setProjection(Projections.rowCount()).uniqueResult

                                    
    posted on 2007-06-08 16:46 Derek.Guo 閱讀(1057) 評論(0)  編輯  收藏 所屬分類: Java
    MSN:envoydada@hotmail.com QQ:34935442
    主站蜘蛛池模板: 亚洲视频在线观看免费视频| 亚洲午夜精品一区二区| 亚洲欧美乱色情图片| 亚洲第一视频网站| 免费无码又爽又刺激网站直播| 最近最新MV在线观看免费高清| 亚洲天堂中文资源| 91香蕉国产线观看免费全集| 久久亚洲精品无码VA大香大香| 99re免费在线视频| 亚洲日韩区在线电影| 色欲国产麻豆一精品一AV一免费| 亚洲视频一区网站| 日韩毛片免费无码无毒视频观看| 亚洲日本VA午夜在线影院| 日本一线a视频免费观看| 精品在线免费视频| 夜夜春亚洲嫩草影院| 亚洲youwu永久无码精品| 国产婷婷高清在线观看免费| 一个人看的www免费高清| 亚洲AV无码一区二区二三区软件| 最近2022中文字幕免费视频| 亚洲精品天堂在线观看| 亚洲精品无码久久不卡| 91在线免费视频| 亚洲一区中文字幕在线观看| 日本黄色免费观看| 91视频免费观看高清观看完整| 久久丫精品国产亚洲av不卡 | 亚洲性日韩精品一区二区三区| 亚洲AV无码乱码在线观看代蜜桃 | 免费看搞黄视频网站| 亚洲国产超清无码专区| 一区二区三区亚洲视频| 亚洲精品久久无码| 久久久久亚洲精品天堂久久久久久| 久久午夜伦鲁片免费无码| 精品国产日韩久久亚洲| 国产精品亚洲精品日韩已方| 国产成人福利免费视频|