各種Hibernate保存方式的區(qū)別
Hibernate保存方式有很多,他們之間有很多不同,這里細(xì)說一下,以便區(qū)別。
一、預(yù)備知識
對于Hibernate,它的對象有三種狀態(tài),transient、persistent、detached
下邊是常見的翻譯辦法:
transient:瞬態(tài)或者自由態(tài)
(new DeptPo(1,”行政部”,20,”行政相關(guān)”),該po的實例和session沒有關(guān)聯(lián),該po的實例處于transient)
persistent:持久化狀態(tài)
(和數(shù)據(jù)庫中記錄想影射的Po實例,它的狀態(tài)是persistent, 通過get和load等得到的對象都是persistent)
detached:脫管狀態(tài)或者游離態(tài)
(1)當(dāng)通過get或load方法得到的po對象它們都處于persistent,但如果執(zhí)行delete(po)時(但不能執(zhí)行事務(wù)),該po狀態(tài)就處于detached, (表示和session脫離關(guān)聯(lián)),因delete而變成游離態(tài)可以通過save或saveOrUpdate()變成持久態(tài)
(2)當(dāng)把session關(guān)閉時,session緩存中的persistent的po對象也變成detached,因關(guān)閉session而變成游離態(tài)的可以通過lock、save、update變成持久態(tài),持久態(tài)實例可以通過調(diào)用 delete()變成脫管狀態(tài)。通過get()或load()方法得到的實例都是持久化狀態(tài)的。脫管狀態(tài)的實例可以通過調(diào)用lock()或者replicate()進(jìn)行持久化。
save()和persist()將會引發(fā)SQL的INSERT,delete()會引發(fā)SQLDELETE,而update()或merge()會引發(fā)SQL UPDATE。對持久化(persistent)實例的修改在刷新提交的時候會被檢測到,它也會引起SQL UPDATE。saveOrUpdate()或者replicate()會引發(fā)SQLINSERT或者UPDATE。
有了這些預(yù)備知識,便可以對不同的Hibernate保存方式驚醒比較了。
二、save 和update區(qū)別
把這一對放在***位的原因是因為這一對是最常用的。save的作用是把一個新的對象保存。update是把一個脫管狀態(tài)的對象或自由態(tài)對象(一定要和一個記錄對應(yīng))更新到數(shù)據(jù)庫。
三、update 和saveOrUpdate區(qū)別
這個是比較好理解的,顧名思義,saveOrUpdate基本上就是合成了save和update,而update只是update;引用Hibernate reference中的一段話來解釋他們的使用場合和區(qū)別,通常下面的場景會使用update()或saveOrUpdate():
程序在***個session中加載對象,接著把session關(guān)閉
該對象被傳遞到表現(xiàn)層
對象發(fā)生了一些改動
該對象被返回到業(yè)務(wù)邏輯層最終到持久層
程序創(chuàng)建第二session調(diào)用第二個session的update()方法持久這些改動
saveOrUpdate(po)做下面的事:
如果該po對象已經(jīng)在本session中持久化了,在本session中執(zhí)行saveOrUpdate不做任何事
如果savaOrUpdate(新po)與另一個與本session關(guān)聯(lián)的po對象擁有相同的持久化標(biāo)識(identifier),拋出一個異常
org.Hibernate.NonUniqueObjectException: a different object with the same identifier value was already associated with the session: [org.itfuture.www.po.Xtyhb#5]
saveOrUpdate如果對象沒有持久化標(biāo)識(identifier)屬性,對其調(diào)用save() ,否則update() 這個對象
四、persist和save區(qū)別
這個是最迷離的一對,表面上看起來使用哪個都行,在Hibernate reference文檔中也沒有明確的區(qū)分他們.
這里給出一個明確的區(qū)分。(可以跟進(jìn)src看一下,雖然實現(xiàn)步驟類似,但是還是有細(xì)微的差別)
主要內(nèi)容區(qū)別:
1,persist把一個瞬態(tài)的實例持久化,但是并"不保證"標(biāo)識符(identifier主鍵對應(yīng)的屬性)被立刻填入到持久化實例中,標(biāo)識符的填入可能被推遲到flush的時候。
2,save, 把一個瞬態(tài)的實例持久化標(biāo)識符,及時的產(chǎn)生,它要返回標(biāo)識符,所以它會立即執(zhí)行Sql insert
五、saveOrUpdate,merge和update區(qū)別
比較update和merge,update的作用上邊說了,這里說一下merge的,如果session中存在相同持久化標(biāo)識(identifier)的實例,用用戶給出的對象覆蓋session已有的持久實例:
(1)當(dāng)我們使用update的時候,執(zhí)行完成后,會拋出異常。
(2)但當(dāng)我們使用merge的時候,把處理自由態(tài)的po對象A的屬性copy到session當(dāng)中處于持久態(tài)的po的屬性中,執(zhí)行完成后原來是持久狀態(tài)還是持久態(tài),而我們提供的A還是自由態(tài)。
六、flush和update區(qū)別
這兩個的區(qū)別好理解,update操作的是在自由態(tài)或脫管狀態(tài)(因session的關(guān)閉而處于脫管狀態(tài))的對象//updateSQL
而flush是操作的在持久狀態(tài)的對象。默認(rèn)情況下,一個持久狀態(tài)的對象的改動(包含set容器)是不需要update的,只要你更改了對象的值,等待Hibernate flush就自動更新或保存到數(shù)據(jù)庫了。Hibernate flush發(fā)生在以下幾種情況中:
1,調(diào)用某些查詢的和手動flush(),session的關(guān)閉、SessionFactory關(guān)閉結(jié)合get()一個對象,把對象的屬性進(jìn)行改變,把資源關(guān)閉。
2,transaction commit的時候(包含了flush)
七、lock和update區(qū)別
update是把一個已經(jīng)更改過的脫管狀態(tài)的對象變成持久狀態(tài),lock是把一個沒有更改過的脫管狀態(tài)的對象變成持久狀態(tài)(針對的是因Session的關(guān)閉而處于脫管狀態(tài)的po對象(2),不能針對因delete而處于脫管狀態(tài)的po對象)
對應(yīng)更改一個記錄的內(nèi)容,兩個的操作不同:
update的操作步驟是:
(1)屬性改動后的脫管的對象的修改->調(diào)用update
lock的操作步驟是:
(2)調(diào)用lock把未修改的對象從脫管狀態(tài)變成持久狀態(tài)-->更改持久狀態(tài)的對象的內(nèi)容-->等待flush或者手動flush
八、clear和evcit的區(qū)別
clear完整的清除session緩存
evcit(obj)把某個持久化對象從session的緩存中清空。
以上便是對比較難以區(qū)分的不同的Hibernate保存方式進(jìn)行的比較。
【編輯推薦】