持久化層的Java對象可處於哪些狀態?這些狀態有哪些特徵?

當應用程序通過new語句創建了一個對象,這個對象的生命週期就開始了,當不再有任何引用變量引用它,這個對象就結束生命週期,它佔用的內存就可以被JVM的垃圾回收器回收。對於需要被持久化的Java對象,在它的生命週期中,可處於以下三個狀態之一:
(1) 臨時狀態(transient):剛剛用new語句創建,還沒有被持久化,不處於Session的緩存中。處於臨時狀態的Java對象被稱爲臨時對象。
(2) 持久化狀態(persistent):已經被持久化,加入到Session的緩存中。處於持久化狀態的Java對象被稱爲持久化對象。
(3) 遊離狀態(detached):已經被持久化,但不再處於Session的緩存中。處於遊離狀態的Java對象被稱爲遊離對象。


圖1爲Java對象的完整狀態轉換圖,Session的特定方法觸發Java對象由一個狀態轉換到另一個狀態。從圖1看出,當Java對象處於臨時狀態或遊離狀態,只要不被任何變量引用,就會結束生命週期,它佔用的內存就可以被JVM的垃圾回收器回收;當處於持久化狀態,由於Session的緩存會引用它,因此它始終處於生命週期中。
臨時對象的特徵
臨時對象具有以下特徵:
(1) 不處於Session的緩存中,也可以說,不被任何一個Session實例關聯。
(2) 在數據庫中沒有對應的記錄。
在以下情況下,Java對象進入臨時狀態:
(1) 當通過new語句剛創建了一個Java對象,它處於臨時狀態,此時不和數據庫中的任何記錄對應。
(2) Session的delete()方法能使一個持久化對象或遊離對象轉變爲臨時對象。對於遊離對象,delete()方法從數據庫中刪除與它對應的記錄;對於持久化對象,delete()方法從數據庫中刪除與它對應的記錄,並且把它從Session的緩存中刪除。
持久化對象的特徵
持久化對象具有以下特徵:
(1) 位於一個Session實例的緩存中,也可以說,持久化對象總是被一個Session實例關聯。
(2) 持久化對象和數據庫中的相關記錄對應。
(3) Session在清理緩存時,會根據持久化對象的屬性變化,來同步更新數據庫。
Session的許多方法都能夠觸發Java對象進入持久化狀態:
(1) Session的save()方法把臨時對象轉變爲持久化對象。
(2) Session的load()或get()方法返回的對象總是處於持久化狀態。
(3) Session的find()方法返回的List集合中存放的都是持久化對象。
(4) Session的update()、saveOrUpdate()和lock()方法使遊離對象轉變爲持久化對象。(nate注:根據hibernate reference的說法當試圖用update更新一個持久化對象時會拋異常)
(5)當一個持久化對象關聯一個臨時對象,在允許級聯保存的情況下,Session在清理緩存時會把這個臨時對象也轉變爲持久化對象。
遊離對象的特徵
遊離對象具有以下特徵:
(1) 不再位於Session的緩存中,也可以說,遊離對象不被Session關聯。
(2) 遊離對象是由持久化對象轉變過來的,因此在數據庫中可能還存在與它對應的記錄(前提條件是沒有其他程序刪除了這條記錄)。
遊離對象與臨時對象的相同之處在於,兩者都不被Session關聯,因此Hibernate不會保證它們的屬性變化與數據庫保持同步。遊離對象與臨時對象的區別在於:前者是由持久化對象轉變過來的,因此可能在數據庫中還存在對應的記錄,而後者在數據庫中沒有對應的記錄。
Session的以下方法使持久化對象轉變爲遊離對象:
(1) 當調用Session的close()方法時,Session的緩存被清空,緩存中的所有持久化對象都變爲遊離對象。如果在應用程序中沒有引用變量引用這些遊離對象,它們就會結束生命週期。

(2)Session的evict()方法能夠從緩存中刪除一個持久化對象,使它變爲遊離狀態。當Session的緩存中保存了大量的持久化對象,會消耗許多內存空間,爲了提高性能,可以考慮調用evict()方法,從緩存中刪除一些持久化對象。但是多數情況下不推薦使用evict()方法,而應該通過查詢語言,或者顯式的導航來控制對象圖的深度

轉載自 :http://blog.sina.com.cn/s/blog_49d3ec2f0102dven.html


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章