Hibernate傳播性持久化攻略
本文向大家介紹Hibernate傳播性持久化(transitive persistence),可能好多人還不了解Hibernate傳播性持久化(transitive persistence),沒(méi)有關(guān)系,看完本文你肯定有不少收獲,希望本文能教會(huì)你更多東西。
對(duì)每一個(gè)對(duì)象都要執(zhí)行保存,刪除或重關(guān)聯(lián)操作讓人感覺(jué)有點(diǎn)麻煩,尤其是在處理許多彼此關(guān)聯(lián)的對(duì)象的時(shí)候。 一個(gè)常見(jiàn)的例子是父子關(guān)系。考慮下面的例子:
如果一個(gè)父子關(guān)系中的子對(duì)象是值類型(value typed)(例如,地址或字符串的集合)的,他們的生命周期會(huì)依賴于父對(duì)象,可以享受方便的級(jí)聯(lián)操作(Cascading),不需要額外的動(dòng)作。
父對(duì)象被保存時(shí),這些值類型(value typed)子對(duì)象也將被保存;父對(duì)象被刪除時(shí),子對(duì)象也將被刪除。 這對(duì)將一個(gè)子對(duì)象從集合中移除是同樣有效:Hibernate會(huì)檢測(cè)到,并且因?yàn)橹殿愋?value typed)的對(duì)象不可能被其他對(duì)象引用,所以Hibernate會(huì)在數(shù)據(jù)庫(kù)中刪除這個(gè)子對(duì)象。
現(xiàn)在考慮同樣的場(chǎng)景,不過(guò)父子對(duì)象都是實(shí)體(entities)類型,而非值類型(value typed)(例如,類別與個(gè)體,或母貓和小貓)。 實(shí)體有自己的生命期,允許共享對(duì)其的引用(因此從集合中移除一個(gè)實(shí)體,不意味著它可以被刪除), 并且實(shí)體到其他關(guān)聯(lián)實(shí)體之間默認(rèn)沒(méi)有級(jí)聯(lián)操作的設(shè)置。Hibernate默認(rèn)不實(shí)現(xiàn)所謂的可到達(dá)即Hibernate傳播性持久化(persistence by reachability)的策略。
每個(gè)Hibernate session的基本操作 - 包括 persist(), merge(), saveOrUpdate(), delete(), lock(), refresh(), evict(), replicate() - 都有對(duì)應(yīng)的級(jí)聯(lián)風(fēng)格(cascade style)。 這些級(jí)聯(lián)風(fēng)格(cascade style)風(fēng)格分別命名為 create, merge, save-update, delete, lock, refresh, evict, replicate。 如果你希望一個(gè)操作被順著關(guān)聯(lián)關(guān)系級(jí)聯(lián)傳播,你必須在映射文件中指出這一點(diǎn)。例如:
- <one-to-one name="person" cascade="persist"/>
級(jí)聯(lián)風(fēng)格(cascade style)是可組合的:
- <one-to-one name="person" cascade="persist,delete,lock"/>
你可以使用cascade="all"來(lái)指定全部操作都順著關(guān)聯(lián)關(guān)系級(jí)聯(lián)(cascaded)。 默認(rèn)值是cascade="none",即任何操作都不會(huì)被級(jí)聯(lián)(cascaded)。
注意有一個(gè)特殊的級(jí)聯(lián)風(fēng)格(cascade style) delete-orphan,只應(yīng)用于one-to-many關(guān)聯(lián),表明delete()操作 應(yīng)該被應(yīng)用于所有從關(guān)聯(lián)中刪除的對(duì)象。
建議:
通常在
如果子對(duì)象的壽命限定在父親對(duì)象的壽命之內(nèi),可通過(guò)指定cascade="all,delete-orphan"將其變?yōu)樽詣?dòng)生命周期管理的對(duì)象(lifecycle object)。
其他情況,你可根本不需要級(jí)聯(lián)(cascade)。但是如果你認(rèn)為你會(huì)經(jīng)常在某個(gè)事務(wù)中同時(shí)用到父對(duì)象與子對(duì)象,并且你希望少打點(diǎn)兒字,可以考慮使用cascade="persist,merge,save-update"。
可以使用cascade="all"將一個(gè)關(guān)聯(lián)關(guān)系(無(wú)論是對(duì)值對(duì)象的關(guān)聯(lián),或者對(duì)一個(gè)集合的關(guān)聯(lián))標(biāo)記為父/子關(guān)系的關(guān)聯(lián)。 這樣對(duì)父對(duì)象進(jìn)行save/update/delete操作就會(huì)導(dǎo)致子對(duì)象也進(jìn)行save/update/delete操作。
此外,一個(gè)持久的父對(duì)象對(duì)子對(duì)象的淺引用(mere reference)會(huì)導(dǎo)致子對(duì)象被同步save/update。 不過(guò),這個(gè)隱喻(metaphor)的說(shuō)法并不完整。除非關(guān)聯(lián)是
如果父對(duì)象被persist(),那么所有子對(duì)象也會(huì)被persist()
如果父對(duì)象被merge(),那么所有子對(duì)象也會(huì)被merge()
如果父對(duì)象被save(),update()或 saveOrUpdate(),那么所有子對(duì)象則會(huì)被saveOrUpdate()
如果某個(gè)持久的父對(duì)象引用了瞬時(shí)(transient)或者脫管(detached)的子對(duì)象,那么子對(duì)象將會(huì)被saveOrUpdate()
如果父對(duì)象被刪除,那么所有子對(duì)象也會(huì)被delete()
除非被標(biāo)記為cascade="delete-orphan"(刪除“孤兒”模式,此時(shí)不被任何一個(gè)父對(duì)象引用的子對(duì)象會(huì)被刪除), 否則子對(duì)象失掉父對(duì)象對(duì)其的引用時(shí),什么事也不會(huì)發(fā)生。 如果有特殊需要,應(yīng)用程序可通過(guò)顯式調(diào)用delete()刪除子對(duì)象。
***,注意操作的級(jí)聯(lián)可能是在調(diào)用期(call time)或者寫入期(flush time)作用到對(duì)象圖上的。所有的操作,如果允許,都在操作被執(zhí)行的時(shí)候級(jí)聯(lián)到可觸及的關(guān)聯(lián)實(shí)體上。然而,save-upate和delete-orphan是在Session flush的時(shí)候才作用到所有可觸及的被關(guān)聯(lián)對(duì)象上的。
本文只是大概的說(shuō)明了Hibernate傳播性持久化的應(yīng)用,可能還有不足的地方,讀者可以查更多的書去了解。
【編輯推薦】