Hibernae 的延遲加載

Hibernae 的延遲加載是一個非常常用的技術,實體的集合屬性默認會被延遲加載,實體所關聯的實體默認也會被延遲加載。Hibernate 通過這種延遲加載來降低系統的內存開銷,從而保證 Hibernate 的運行性能。

下面先來剖析 Hibernate 延遲加載的“秘密”。

集合屬性的延遲加載

當 Hibernate 從數據庫中初始化某個持久化實體時,該實體的集合屬性是否隨持久化類一起初始化呢?如果集合屬性里包含十萬,甚至百萬的記錄,在初始化持久化實體的同時,完成所有集合屬性的抓取,將導致性能急劇下降。完全有可能系統只需要使用持久化類集合屬性中的部分記錄,而完全不是集合屬性的全部,這樣,沒有必要一次加載所有的集合屬性。

對于集合屬性,通常推薦使用延遲加載策略。所謂延遲加載就是等系統需要使用集合屬性時才從數據庫裝載關聯的數據。

例如下面 Person 類持有一個集合屬性,該集合屬性里的元素的類型為 Address,該 Person 類的代碼片段如下:

清單 1. Person.java

public class Person

{

// 標識屬性

private Integer id;

// Person 的 name 屬性

private String name;

// 保留 Person 的 age 屬性

private int age;

// 使用 Set 來保存集合屬性

private Set addresses = new HashSet();

// 下面省略了各屬性的 setter 和 getter 方法

...

}

為了讓 Hibernate 能管理該持久化類的集合屬性,程序為該持久化類提供如下映射文件:

清單 2. Person.hbm.xml


從上面映射文件的代碼可以看出,Person 的集合屬性中的 Address 類只是一個普通的 POJO。該 Address 類里包含 detail、zip 兩個屬性。由于 Address 類代碼非常簡單,故此處不再給出該類的代碼。

上面映射文件中 元素里的代碼指定了 lazy="true"(對于 元素來說,lazy="true"是默認值),它指定 Hibernate 會延遲加載集合屬性里 Address 對象。

例如通過如下代碼來加載 ID 為 1 的 Person 實體:

Session session = sf.getCurrentSession();

Transaction tx = session.beginTransaction();

Person p = (Person) session.get(Person.class, 1);? //<1>

System.out.println(p.getName());

上面代碼只是需要訪問 ID 為 1 的 Person 實體,并不想訪問這個 Person 實體所關聯的 Address 對象。此時有兩種情況:

如果不延遲加載,Hibernate 就會在加載 Person 實體對應的數據記錄時立即抓取它關聯的 Address 對象。

如果采用延遲加載,Hibernate 就只加載 Person 實體對應的數據記錄。

很明顯,第二種做法既能減少與數據庫的交互,而且避免了裝載 Address 實體帶來的內存開銷——這也是 Hibernate 默認啟用延遲加載的原因。

現在的問題是,延遲加載到底是如何實現的呢? Hibernate 在加載 Person 實體時,Person 實體的 addresses 屬性值是什么呢?

為了解決這個問題,我們在<1>號代碼處設置一個斷點,在 Eclipse 中進行 Debug,此時可以看到 Eclipse 的 Console 窗口有如圖 1 所示的輸出:

圖 1. 延遲加載集合屬性的 Console 輸出

正如圖 1 輸出所看到的,此時 Hibernate 只從 Person 實體對應的數據表中抓取數據,并未從 Address 對象對應的數據表中抓取數據,這就是延遲加載。

那么 Person 實體的 addresses 屬性是什么呢?此時可以從 Eclipse 的 Variables 窗口看到如圖 2 所示的結果:

圖 2. 延遲加載的集合屬性值

從圖 2 的方框里的內容可以看出,這個 addresses 屬性并不是我們熟悉的 HashSet、TreeSet 等實現類,而是一個 PersistentSet 實現類,這是 Hibernate 為 Set 接口提供的一個實現類。

PersistentSet 集合對象并未真正抓取底層數據表的數據,因此自然也無法真正去初始化集合里的 Address 對象。不過 PersistentSet 集合里持有一個 session 屬性,這個 session 屬性就是 Hibernate Session,當程序需要訪問 PersistentSet 集合元素時,PersistentSet 就會利用這個 session 屬性去抓取實際的 Address 對象對應的數據記錄。

那么到底抓取那些 Address 實體對應的數據記錄呢?這也難不倒 PersistentSet,因為 PersistentSet 集合里還有一個 owner 屬性,該屬性就說明了 Address 對象所屬的 Person 實體,Hibernate 就會去查找 Address 對應數據表中外鍵值參照到該 Person 實體的數據。

例如我們單擊圖 2 所示窗口中 addresses 行,也就是告訴 Eclipse 要調試、輸出 addresses 屬性,這就是要訪問 addresses 屬性了,此時就可以在 Eclipse 的 Console 窗口看到輸出如下 SQL 語句:

select

addresses0_.person_id as person1_0_0_,

addresses0_.detail as detail0_,

addresses0_.zip as zip0_

from

person_address addresses0_

where

addresses0_.person_id=?

這就是 PersistentSet 集合跟據 owner 屬性去抓取特定 Address 記錄的 SQL 語句。此時可以從 Eclipse 的 Variables 窗口看到圖 3 所示的輸出:

圖 3. 已加載的集合屬性值

從圖 3 可以看出,此時的 addresses 屬性已經被初始化了,集合里包含了 2 個 Address 對象,這正是 Person 實體所關聯的兩個 Address 對象。

通過上面介紹可以看出,Hibernate 對于 Set 屬性延遲加載關鍵就在于 PersistentSet 實現類。在延遲加載時,開始 PersistentSet 集合里并不持有任何元素。但 PersistentSet 會持有一個 Hibernate Session,它可以保證當程序需要訪問該集合時“立即”去加載數據記錄,并裝入集合元素。

與 PersistentSet 實現類類似的是,Hibernate 還提供了 PersistentList、PersistentMap、PersistentSortedMap、PersistentSortedSet 等實現類,它們的功能與 PersistentSet 的功能大致類似。

熟悉 Hibernate 集合屬性讀者應該記得:Hibernate 要求聲明集合屬性只能用 Set、List、Map、SortedSet、SortedMap 等接口,而不能用 HashSet、ArrayList、HashMap、TreeSet、TreeMap 等實現類,其原因就是因為 Hibernate 需要對集合屬性進行延遲加載,而 Hibernate 的延遲加載是依靠 PersistentSet、PersistentList、PersistentMap、PersistentSortedMap、PersistentSortedSet 來完成的——也就是說,Hibernate 底層需要使用自己的集合實現類來完成延遲加載,因此它要求開發者必須用集合接口、而不是集合實現類來聲明集合屬性。

Hibernate 對集合屬性默認采用延遲加載,在某些特殊的情況下,為 、、 等元素設置 lazy="false"屬性來取消延遲加載。

回頁首

關聯實體的延遲加載

默認情況下,Hibernate 也會采用延遲加載來加載關聯實體,不管是一對多關聯、還是一對一關聯、多對多關聯,Hibernate 默認都會采用延遲加載。

對于關聯實體,可以將其分為兩種情況:

關聯實體是多個實體時(包括一對多、多對多):此時關聯實體將以集合的形式存在,Hibernate 將使用 PersistentSet、PersistentList、PersistentMap、PersistentSortedMap、PersistentSortedSet 等集合來管理延遲加載的實體。這就是前面所介紹的情形。

關聯實體是單個實體時(包括一對一、多對一):當 Hibernate 加載某個實體時,延遲的關聯實體將是一個動態生成代理對象。

當關聯實體是單個實體時,也就是使用 或 映射關聯實體的情形,這兩個元素也可通過 lazy 屬性來指定延遲加載。

下面例子把 Address 類也映射成持久化類,此時 Address 類也變成實體類,Person 實體與 Address 實體形成一對多的雙向關聯。此時的映射文件代碼如下:

清單 3. Person.hbm.xml

原文出處:http://blog.csdn.net/xc635960736/article/details/7049863

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,698評論 6 539
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,202評論 3 426
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事?!?“怎么了?”我有些...
    開封第一講書人閱讀 177,742評論 0 382
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,580評論 1 316
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,297評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,688評論 1 327
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,693評論 3 444
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,875評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,438評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,183評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,384評論 1 372
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,931評論 5 363
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,612評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,022評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,297評論 1 292
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,093評論 3 397
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,330評論 2 377

推薦閱讀更多精彩內容

  • 這就是 PersistentSet 集合跟據 owner 屬性去抓取特定 Address 記錄的 SQL 語句。此...
    FTOLsXD閱讀 443評論 0 1
  • 什么是懶加載?他的作用? 延遲加載,也叫懶加載,它是hibernate為提高程序執行效率而提供的一種機制,即只有真...
    FTOLsXD閱讀 251評論 0 1
  • Spring Cloud為開發人員提供了快速構建分布式系統中一些常見模式的工具(例如配置管理,服務發現,斷路器,智...
    卡卡羅2017閱讀 134,816評論 18 139
  • 摘自:《大慧普覺禪師法語》 近世學者,多棄本逐末,背正投邪,只以為學、為道、為名,專以取富貴張大門戶為決定義,故心...
    了義讀經閱讀 271評論 0 0
  • 我笑出了聲 當跳上臺階 身上的雨珠也跟著蹦了下來 “追不到我了吧” 轉身看著身后地面上跳腳的水花 剛才在雨中奔跑的...
    慧海安地閱讀 306評論 0 1