Hibernate DetachedCriteria學(xué)習(xí)經(jīng)驗(yàn)
本文向大家介紹Hibernate DetachedCriteria,可能好多人還不了解,沒(méi)有關(guān)系,看完本文你肯定有不少收獲,希望本文能教會(huì)你更多東西。
Hibernate DetachedCriteria,這是一個(gè)非常有意義的特性!我們知道,在常規(guī)的Web編程中,有大量的動(dòng)態(tài)條件查詢,即用戶在網(wǎng)頁(yè)上面自由選擇某些條件,程序根據(jù)用戶的選擇條件,動(dòng)態(tài)生成SQL語(yǔ)句,進(jìn)行查詢。
針對(duì)這種需求,對(duì)于分層應(yīng)用程序來(lái)說(shuō),Web層需要傳遞一個(gè)查詢的條件列表給業(yè)務(wù)層對(duì)象,業(yè)務(wù)層對(duì)象獲得這個(gè)條件列表之后,然后依次取出條件,構(gòu)造查詢語(yǔ)句。這里的一個(gè)難點(diǎn)是條件列表用什么來(lái)構(gòu)造?傳統(tǒng)上使用Map,但是這種方式缺陷很大,Map可以傳遞的信息非常有限,只能傳遞name和value,無(wú)法傳遞究竟要做怎樣的條件運(yùn)算,究竟是大于,小于,like,還是其它的什么,業(yè)務(wù)層對(duì)象必須確切掌握每條entry的隱含條件。因此一旦隱含條件改變,業(yè)務(wù)層對(duì)象的查詢構(gòu)造算法必須相應(yīng)修改,但是這種查詢條件的改變是隱式約定的,而不是程序代碼約束的,因此非常容易出錯(cuò)。
DetachedCriteria可以解決這個(gè)問(wèn)題,即在web層,程序員使用DetachedCriteria來(lái)構(gòu)造查詢條件,然后將這個(gè)DetachedCriteria作為方法調(diào)用參數(shù)傳遞給業(yè)務(wù)層對(duì)象。而業(yè)務(wù)層對(duì)象獲得DetachedCriteria之后,可以在session范圍內(nèi)直接構(gòu)造Criteria,進(jìn)行查詢。就此,查詢語(yǔ)句的構(gòu)造完全被搬離到web層實(shí)現(xiàn),而業(yè)務(wù)層則只負(fù)責(zé)完成持久化和查詢的封裝即可,與查詢條件構(gòu)造完全解耦,非常***!這恐怕也是以前很多企圖在web層代碼中構(gòu)造HQL語(yǔ)句的人想實(shí)現(xiàn)的夢(mèng)想吧!
示例代碼片段如下:
- 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是一對(duì)多關(guān)聯(lián),查詢條件為:名稱是“department”開發(fā)部門;部門里面的雇員年齡大于20歲;
業(yè)務(wù)層對(duì)象使用該條件執(zhí)行查詢:
- detachedCriteria.getExecutableCriteria(session).list();
***的意義在于,業(yè)務(wù)層代碼是固定不變的,所有查詢條件的構(gòu)造都在web層完成,業(yè)務(wù)層只負(fù)責(zé)在session內(nèi)執(zhí)行之。這樣代碼就可放之四海而皆準(zhǔn),都無(wú)須修改了。然而Spring和Hibernate DetachedCriteria有不兼容的問(wèn)題,因此在Spring環(huán)境下面使用Hibernate3需要注意:
Spring的HibernateTemplate提供了Hibernate的***封裝,即通過(guò)匿名類實(shí)現(xiàn)回調(diào),來(lái)保證Session的自動(dòng)資源管理和事務(wù)的管理。其中核心方法是:
- HibernateTemplate.execute(new HibernateCallback() {
- public Object doInHibernate(Session session) throws HibernateException {
- ....
- }
- }
回調(diào)方法提供了session作為參數(shù),有了session,就可以自由的使用Hibernate API編程了。使用了spring的之后,代碼修改如下:
- 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);
【編輯推薦】