溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Hibernate中抓取策略是怎么樣的

發(fā)布時間:2021-11-30 15:45:34 來源:億速云 閱讀:118 作者:小新 欄目:編程語言

小編給大家分享一下Hibernate中抓取策略是怎么樣的,相信大部分人都還不怎么了解,因此分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后大有收獲,下面讓我們一起去了解一下吧!

Hibernate抓取策略(fetching strategy)是指:當(dāng)應(yīng)用程序需要在(Hibernate實體對象圖的)關(guān)聯(lián)關(guān)系間進(jìn)行導(dǎo)航的時候, Hibernate如何獲取關(guān)聯(lián)對象的策略。

抓取策略可以在O/R映射的元數(shù)據(jù)中聲明,也可以在特定的HQL 或條件查詢(Criteria Query)中重載聲明。

如下幾種Hibernate抓取策略

◆連接抓?。↗oin fetching) - Hibernate通過 在SELECT語句使用OUTER JOIN(外連接)來 獲得對象的關(guān)聯(lián)實例或者關(guān)聯(lián)集合。

◆查詢抓?。⊿elect fetching) - 另外發(fā)送一條 SELECT 語句抓取當(dāng)前對象的關(guān)聯(lián)實體或集合。除非你顯式的指定lazy="false"禁止 延遲抓取(lazy fetching),否則只有當(dāng)你真正訪問關(guān)聯(lián)關(guān)系的時候,才會執(zhí)行第二條select語句。

◆子查詢抓?。⊿ubselect fetching) - 另外發(fā)送一條SELECT 語句抓取在前面查詢到(或者抓取到)的所有實體對象的關(guān)聯(lián)集合。除非你顯式的指定lazy="false" 禁止延遲抓?。╨azy fetching),否則只有當(dāng)你真正訪問關(guān)聯(lián)關(guān)系的時候,才會執(zhí)行第二條select語句。

◆批量抓取(Batch fetching) - 對查詢抓取的優(yōu)化方案, 通過指定一個主鍵或外鍵列表,Hibernate使用單條SELECT語句獲取一批對象實例或集合。

Hibernate抓取策略會區(qū)分下列各種情況:

1.Immediate fetching,立即抓取 - 當(dāng)宿主被加載時,關(guān)聯(lián)、集合或?qū)傩员涣⒓醋ト ?/p>

2.Lazy collection fetching,延遲集合抓取- 直到應(yīng)用程序?qū)线M(jìn)行了一次操作時,集合才被抓取。(對集合而言這是默認(rèn)行為。)

3."Extra-lazy" collection fetching,"Extra-lazy"集合抓取 -對集合類中的每個元素而言,都是直到需要時才去訪問數(shù)據(jù)庫。除非絕對必要,Hibernate不會試圖去把整個集合都抓取到內(nèi)存里來(適用于非常大的集合)。

4.Proxy fetching,代理抓取 - 對返回單值的關(guān)聯(lián)而言,當(dāng)其某個方法被調(diào)用,而非對其關(guān)鍵字進(jìn)行g(shù)et操作時才抓取。

5."No-proxy" fetching,非代理抓取 - 對返回單值的關(guān)聯(lián)而言,當(dāng)實例變量被訪問的時候進(jìn)行抓取。與上面的代理抓取相比,這種方法沒有那么“延遲”得厲害(就算只訪問標(biāo)識符,也會導(dǎo)致關(guān)聯(lián)抓取)但是更加透明,因為對應(yīng)用程序來說,不再看到proxy。這種方法需要在編譯期間進(jìn)行字節(jié)碼增強操作,因此很少需要用到。

6.Lazy attribute fetching,屬性延遲加載 - 對屬性或返回單值的關(guān)聯(lián)而言,當(dāng)其實例變量被訪問的時候進(jìn)行抓取。需要編譯期字節(jié)碼強化,因此這一方法很少是必要的。

這里有兩個正交的概念:關(guān)聯(lián)何時被抓取,以及被如何抓取(會采用什么樣的SQL語句)。不要混淆它們!我們使用抓取來改善性能。我們使用延遲來定義一些契約,對某特定類的某個脫管的實例,知道有哪些數(shù)據(jù)是可以使用的。

1.操作延遲加載的關(guān)聯(lián)

默認(rèn)情況下,Hibernate 3對集合使用延遲select抓取,對返回單值的關(guān)聯(lián)使用延遲代理抓取。對幾乎是所有的應(yīng)用而言,其絕大多數(shù)的關(guān)聯(lián),這種策略都是有效的。

注意:假若你設(shè)置了hibernate.default_batch_fetch_size,Hibernate會對延遲加載采取批量抓取優(yōu)化措施(這種優(yōu)化也可能會在更細(xì)化的級別打開)。

然而,你必須了解延遲抓取帶來的一個問題。在一個打開的Hibernate session上下文之外調(diào)用延遲集合會導(dǎo)致一次意外。比如:

s = sessions.openSession();  Transaction tx = s.beginTransaction();                User u = (User) s.createQuery("from User u where u.name=:userName")      .setString("userName", userName).uniqueResult();  Map permissions = u.getPermissions();   tx.commit();  s.close();   Integer accessLevel = (Integer) permissions.get("accounts");  // Error!

在Session關(guān)閉后,permessions集合將是未實例化的、不再可用,因此無法正常載入其狀態(tài)。 Hibernate對脫管對象不支持延遲實例化. 這里的修改方法是:將permissions讀取數(shù)據(jù)的代碼 移到tx.commit()之前。

除此之外,通過對關(guān)聯(lián)映射指定lazy="false",我們也可以使用非延遲的集合或關(guān)聯(lián)。但是, 對絕大部分集合來說,更推薦使用延遲方式抓取數(shù)據(jù)。如果在你的對象模型中定義了太多的非延遲關(guān)聯(lián),Hibernate最終幾乎需要在每個事務(wù)中載入整個數(shù)據(jù)庫到內(nèi)存中!

但是,另一方面,在一些特殊的事務(wù)中,我們也經(jīng)常需要使用到連接抓?。ㄋ旧砩暇褪欠茄舆t的),以代替查詢抓取。 下面我們將會很快明白如何具體的定制Hibernate中的抓取策略。在Hibernate3中,具體選擇哪種抓取策略的機制是和選擇 單值關(guān)聯(lián)或集合關(guān)聯(lián)相一致的。

2.  調(diào)整抓取策略(Tuning fetch strategies)

查詢抓取(默認(rèn)的)在N+1查詢的情況下是極其脆弱的,因此我們可能會要求在映射文檔中定義使用連接抓?。?/p>

<set name="permissions"               fetch="join">     <key column="userId"/>     <one-to-many class="Permission"/> set <many-to-one name="mother" class="Cat" fetch="join"/>

在映射文檔中定義的抓取策略將會對以下列表條目產(chǎn)生影響:通過get()或load()方法取得數(shù)據(jù)。只有在關(guān)聯(lián)之間進(jìn)行導(dǎo)航時,才會隱式的取得數(shù)據(jù)。

條件查詢,使用了subselect抓取的HQL查詢

不管你使用哪種抓取策略,定義為非延遲的類圖會被保證一定裝載入內(nèi)存。注意這可能意味著在一條HQL查詢后緊跟著一系列的查詢。

通常情況下,我們并不使用映射文檔進(jìn)行抓取策略的定制。更多的是,保持其默認(rèn)值,然后在特定的事務(wù)中, 使用HQL的左連接抓?。╨eft join fetch) 對其進(jìn)行重載。這將通知 Hibernate在***次查詢中使用外部關(guān)聯(lián)(outer join),直接得到其關(guān)聯(lián)數(shù)據(jù)。 在條件查詢 API中,應(yīng)該調(diào)用 setFetchMode(FetchMode.JOIN)語句。

也許你喜歡僅僅通過條件查詢,就可以改變get() 或 load()語句中的數(shù)據(jù)抓取策略。例如:

User user = (User) session.createCriteria(User.class)                  .setFetchMode("permissions", FetchMode.JOIN)                  .add( Restrictions.idEq(userId) )                  .uniqueResult();


(這就是其他ORM解決方案的“抓取計劃(fetch plan)”在Hibernate中的等價物。)截然不同的一種避免N+1次查詢的方法是,使用二級緩存。

以上是“Hibernate中抓取策略是怎么樣的”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對大家有所幫助,如果還想學(xué)習(xí)更多知識,歡迎關(guān)注億速云行業(yè)資訊頻道!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI