Hibernate管理Session和批量操作分析
本文詳細(xì)分析了Hibernate管理Session和批量操作的用法。分享給大家供大家參考。具體分析如下:
Hibernate管理Session
Hibernate自身提供了三種管理Session對象的方法
① Session對象的生命周期與本地線程綁定
② Session對象的生命周期與JTA事務(wù)綁定
③ Hibernate委托程序管理Session對象的生命周期
在Hibernate的配置文件中,hibernate.current_session_context_class屬性用于指定Session管理方式,可選值包括:
① thread:Session對象的生命周期與本地線程綁定
② jta*:Session對象的生命周期與JTA事務(wù)綁定
③ managed:Hibernate委托程序來管理Session對象的生命周期
Session對象的生命周期與本地線程綁定:
如果把Hibernate配置文件的hibernate.current_session_context_class屬性值設(shè)置為thread,Hibernate就會按照與本地線程綁定的方式來管理Session
Hibernate按以下規(guī)則把Session與本地線程綁定:
當(dāng)一個線程(thread)第一次調(diào)用SessionFactory對象的getCurrentSession()方法時,該方法會創(chuàng)建一個新的Session(sessionA)對象,把該對象與threadA綁定,并將session返回
當(dāng)threadA再次調(diào)用SessionFactory對象的getCurrentSession()方法時,該方法將返回sessionA對象
當(dāng)threadA提交sessionA對象關(guān)聯(lián)的事務(wù)時,Hibernate會自動flush sessionA對象的緩存,然后提交事務(wù),關(guān)閉session隨心。當(dāng)threadA撤銷sessionA對象關(guān)聯(lián)的事務(wù)時,也會自動關(guān)閉sessionA對象
若threadA再次調(diào)用SessionFactory對象的getCurrentSession()方法時,該方法會又創(chuàng)建一個新的Session(sessionB)對象,把該對象與threadA綁定,并將sessionB返回
批量處理數(shù)據(jù)
批量處理數(shù)據(jù)是指在一個事務(wù)中處理大量數(shù)據(jù)
在應(yīng)用層進程批量操作,主要有以下方式:
① 通過Session
② 通過HQL
③ 通過StatelessSession
④ 通過JDBC API----推薦此種,因為速度最快
Session進行批量操作:
Session的save()及update()方法都會把處理的對象存放在自己的緩存中。如果通過一個Session對象來處理大量持久化對象,應(yīng)該及時從緩存中清空已經(jīng)處理完畢并且不會再訪問的對象。具體的做法是在處理完一個對象或小批量對象后,立即調(diào)用flush()方法刷新緩存,然后再調(diào)用clear()方法情況緩存
通過Session來進行處理操作會受到以下約束:
需要在Hibernate配置文件中設(shè)置JDBC單次批量處理的數(shù)目,應(yīng)保證每次向數(shù)據(jù)庫發(fā)送的批量的SQL語句數(shù)目與batch size屬性一致
若對象采用"identity"標(biāo)識生成器,則Hibernate無法在JDBC曾進行批量插入操作
進行批量操作時,建議關(guān)閉Hibernate的二級緩存
批量插入數(shù)據(jù)代碼演示:
for(int i = 0; i < 10000; i++) {
news = new News();
news.setTitle("--" + i);
session.save(news);
if((i + 1) % 20 == 0) {
session.flush();
session.clear();
}
}
批量更新:在進行批量更新時,如果一下子把所有對象都加載到Session緩存,然后再緩存中一一更新,顯然是不可取的
使用可滾動的結(jié)果集org.hibernate.ScrollableResults,該對象中實際上并不包含任何對象,只包含用于在線定位記錄的游標(biāo)。只有當(dāng)程序遍歷訪問ScrollableResults對象的特定元素時,它才會到數(shù)據(jù)庫中加載相應(yīng)的對象
org.hibernate.ScrollableResults對象由Query的scroll方法返回
通過HQL進行批量操作:
注意:HQL只支持INSERT INTO ... SELECT形式的插入語句,但不支持INSERT INTO ... VALUES形式的插入語句。所以使用HQL不能進行批量插入操作
通過StatelessSession進行批量操作:
從形式上看,StatelessSession與Session的用法類似。StatelessSession與Session相比,有以下區(qū)別:
StatelessSession沒有緩存,通過StatelessSession來加載、保存或更新后的對象處于游離狀態(tài)
StatelessSession不會與Hibernate的二級緩存交互
當(dāng)調(diào)用StatelessSession的save()、update()或delete()方法時,這些方法會立即執(zhí)行相應(yīng)的SQL語句,而不會僅計劃執(zhí)行一條SQL語句
StatelessSession不會進行臟檢查,因此修改了Customer對象屬性后,還需要調(diào)用StatelessSession的update()方法來更新數(shù)據(jù)庫中數(shù)據(jù)
StatelessSession不會對關(guān)聯(lián)的對象進行任何的級聯(lián)操作
通過同一個StatelessSession對象兩次加載的OID為1的Customer對象,得到的兩個對象內(nèi)存地址不同
StatelessSession所做的操作可以被Interceptor攔截器捕獲到,但是會被Hibernate的事件處理系統(tǒng)忽略掉
希望本文所述對大家的Java程序設(shè)計有所幫助。
相關(guān)文章
spring-boot-maven-plugin報紅解決方案(親測有效)
本文主要介紹了spring-boot-maven-plugin報紅解決方案,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-03-03Java的@Transactional、@Aysnc、事務(wù)同步問題詳解
這篇文章主要介紹了Java的@Transactional、@Aysnc、事務(wù)同步問題詳解,現(xiàn)在我們需要在一個業(yè)務(wù)方法中插入一個用戶,這個業(yè)務(wù)方法我們需要加上事務(wù),然后插入用戶后,我們要異步的方式打印出數(shù)據(jù)庫中所有存在的用戶,需要的朋友可以參考下2023-11-11java原裝代碼完成pdf在線預(yù)覽和pdf打印及下載
本文主要介紹了java原裝代碼完成pdf在線預(yù)覽和pdf打印及下載的方法,具有一定的參考價值,下面跟著小編一起來看下吧2017-02-02java Socket編程實現(xiàn)I/O多路復(fù)用的示例
本文主要介紹了java Socket編程實現(xiàn)I/O多路復(fù)用的示例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-09-09spring/springboot整合curator遇到的坑及解決
這篇文章主要介紹了spring/springboot整合curator遇到的坑及解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-05-05