memge和saveOrUpdate的區(qū)別
今天做hibernate開發(fā)的時候遇到這樣一個錯誤:
a different object with the same identifier value was already associated with the session
(不同對象具有相同的標(biāo)識符值已經(jīng)與會話關(guān)聯(lián)起來)
牽涉出了Hibernate中幾個易混淆方法的區(qū)別:
首先要介紹下Hibernate中的三種狀態(tài)
Hibernate的對象有3種狀態(tài),分別為:瞬時態(tài)(Transient)、持久態(tài)(Persistent)、脫管態(tài)(Detached)。處于持久態(tài) 的對象也稱為PO(Persistence Object),瞬時對象和脫管對象也稱為VO(Value Object)。
瞬時態(tài)由new命令開辟內(nèi)存空間的java對象,eg. Person person = new Person("xxx", "xx");如
果沒有變量對該對象進(jìn)行引用,它將被java虛擬機(jī)回收。瞬時對象在內(nèi)存孤立存在,它是攜帶信息的載體,不和數(shù)據(jù)庫的數(shù)據(jù)有任何關(guān)聯(lián)關(guān)系,在
Hibernate中,可通過session的save()或 saveOrUpdate()方法將瞬時對象與數(shù)據(jù)庫相關(guān)聯(lián),并將數(shù)據(jù)對應(yīng)的插入數(shù)據(jù)庫
中,此時該瞬時對象轉(zhuǎn)變成持久化對象。
持久態(tài)處于該狀態(tài)的對象在數(shù)據(jù)庫中具有對應(yīng)的記錄,并擁有一個持久化標(biāo)識。如果是
用hibernate的delete()方法,對應(yīng)的持久對象就變成瞬時對象, 因數(shù)據(jù)庫中的對應(yīng)數(shù)據(jù)已被刪除,該對象不再與數(shù)據(jù)庫的記錄關(guān)聯(lián)。當(dāng)一個
session執(zhí)行close()或clear()、evict()之后,持久對象變成脫管對象,此時持久對象會變成脫管對象,此時該對象雖然具有 數(shù)據(jù)
庫識別值,但它已不在HIbernate持久層的管理之下。
持久對象具有如下特點(diǎn):
1.、和session實(shí)例關(guān)聯(lián);
2.、在數(shù)據(jù)庫中有與之關(guān)聯(lián)的記錄。
脫管態(tài)
脫管態(tài)當(dāng)與某持久對象關(guān)聯(lián)的session被關(guān)閉后,該持久對象轉(zhuǎn)變?yōu)槊摴軐ο蟆.?dāng)脫管對象被重新關(guān)聯(lián)到session上時,并再次轉(zhuǎn)變成持久對象。脫管對象擁有數(shù)據(jù)庫的識別值,可通過update()、saveOrUpdate()等方法,轉(zhuǎn)變成持久對象。
脫管對象擁有數(shù)據(jù)庫的識別值,可通過update()、saveOrUpdate()等方法,轉(zhuǎn)變成持久對象。
脫管對象具有如下特點(diǎn):
1. 本質(zhì)上與瞬時對象相同,在沒有任何變量引用它時,JVM會在適當(dāng)?shù)臅r候?qū)⑺厥眨?/p>
2. 比瞬時對象多了一個數(shù)據(jù)庫記錄標(biāo)識值。
下來講下我對merge和saveOrUpdate方法區(qū)別的理解:merge方法是把我們提供的對象轉(zhuǎn)變?yōu)橥泄軤顟B(tài)的對象;而saveOrUpdate則是把我們提供的對象變成一個持久化對象;說的通俗一點(diǎn)就是:saveOrUpdate后的對象會納入session的管理,對象的狀態(tài)會跟數(shù)據(jù)庫同步,再次查詢該對象會直接從session中取,merge后的對 象不會納入session的管理,再次查詢該對象還是會從數(shù)據(jù)庫中取。
所以遇到“a different object with the same identifier value was already associated with the session”這個問題只需要清空session或者將saveOrUpdate方法變?yōu)閙erge方法就可以了。
浙公網(wǎng)安備 33010602011771號