Hibernate使用批量抓取技巧
大家都知道Hibernate可以充分有效的使用批量抓取,也就是說,如果僅一個(gè)訪問代理(或集合),那么Hibernate將不載入其他未實(shí)例化的代理。
Hibernate批量抓取是延遲查詢抓取的優(yōu)化方案,你可以在兩種Hibernate批量抓取方案之間進(jìn)行選擇:在類級(jí)別和集合級(jí)別。
類/實(shí)體級(jí)別的批量抓取很容易理解。假設(shè)你在運(yùn)行時(shí)將需要面對(duì)下面的問題:你在一個(gè)Session中載入了25個(gè)Cat實(shí)例,每個(gè)Cat實(shí)例都擁有一個(gè)引用成員owner, 其指向Person,而Person類是代理,同時(shí)lazy="true"。
如果你必須遍歷整個(gè)cats集合,對(duì)每個(gè)元素調(diào)用getOwner()方法,Hibernate將會(huì)默認(rèn)的執(zhí)行25次SELECT查詢,得到其owner的代理對(duì)象。這時(shí),你可以通過在映射文件的Person屬性,顯式聲明batch-size,改變其行為:
- <class name="Person" batch-size="10">...</class>
隨之,Hibernate將只需要執(zhí)行三次查詢,分別為10、10、 5。
你也可以在集合級(jí)別定義批量抓取。例如,如果每個(gè)Person都擁有一個(gè)延遲載入的Cats集合,現(xiàn)在,Sesssion中載入了10個(gè)person對(duì)象,遍歷person集合將會(huì)引起10次SELECT查詢,每次查詢都會(huì)調(diào)用getCats()方法。如果你在Person的映射定義部分,允許對(duì)cats批量抓取, 那么,Hibernate將可以預(yù)先抓取整個(gè)集合。
請(qǐng)看例子:
- <class name="Person">
- <set name="cats" batch-size="3">
- ...
- </set>
- </class>
如果整個(gè)的batch-size是3(筆誤?),那么Hibernate將會(huì)分四次執(zhí)行SELECT查詢,按照3、3、3、1的大小分別載入數(shù)據(jù)。這里的每次載入的數(shù)據(jù)量還具體依賴于當(dāng)前Session中未實(shí)例化集合的個(gè)數(shù)。
如果你的模型中有嵌套的樹狀結(jié)構(gòu),例如典型的帳單-原料結(jié)構(gòu)(bill-of-materials pattern),集合的Hibernate批量抓取是非常有用的。(盡管在更多情況下對(duì)樹進(jìn)行讀取時(shí),嵌套集合(nested set)或原料路徑(materialized path)(××) 是更好的解決方法。)
【編輯推薦】