Hibernate中消極式載入和緩衝,hibernate消極式載入

來源:互聯網
上載者:User

Hibernate中消極式載入和緩衝,hibernate消極式載入
什麼是消極式載入?

消極式載入是指當應用程式想要從資料庫擷取對象時(在沒有設定lazy屬性值為false),Hibernate只是從資料庫擷取合格對象的OId從而組建代理程式對象,並沒有載入出對象

訪問該對象的屬性時才會載入出相應的值。簡答來說就是儘可能的減少查詢的資料量。

如何配置消極式載入

在Hibernate中通過.hbm設定檔中的lazy屬性來陪值,並且lazy屬性出現的位置不同其作用和取值也不同。下面來詳細介紹其在不同位置的不同取值和作用

類Class標籤中的lazy:

在類標籤Class中出現的lazy取值有true(預設值 消極式載入)、false(立即載入)兩種取值如下所示:

如果lazy取值為false則表示應用程式從資料庫擷取對象時會立即載入所有屬性值(不包含自訂類型屬性)

以員工表為例測試案例如下:

員工表表結構:

員工實體類:

員工表的hbm對應檔:

測試代碼:

測試結果:

從的結果中我們可以看到emp對象並沒有進行延時載入但是其儲存部門(Dept)對象的引用的屬性並沒有進行載入。

我們再來看看當lazy屬性的值為true時的結果

從我們可以看到lazy的屬性不管是True還是false其結果都是一樣的!這是為什麼呢?Lazy屬性的值為true是不是應該延時載入嗎?

注意:我們再上面編寫測試案例時擷取員工對象是用的get方法而我也在之前的部落格中說過session對象的get方法不支援延時載入他會忽略掉類層級的lazy屬性!我們把get方法換成load方法再來測試。

從中我們可以看到當lazy屬性值為true時Hibernate並不會一次性載入出所有屬性值,只有當程式需要時才去載入從而減少了和資料庫互動的負擔,提升了程式的效能,這也是消極式載入出現的目的!

多對一關聯中的lazy               

如果想要在擷取對象的同時立即載入與之關聯的自訂類型屬性就需要在其多對一配置中設定lazy屬性,在此處lazy屬性的取值為:proxy:消極式載入(預設值)、no-proxy:無代理消極式載入,false:立即載入

比如我們在執行個體一的基礎上添加lazy屬性值為false再來測試:

Set元素中的lazy屬性

我們知道如果對象中存在其他實體的集合則需要在hbm檔案中配置set元素來進行表間的映射,而在set元素中也可以添加lazy屬性其取值為:true:消極式載入(預設值)、false:立即載入、extra:加強延時載入。

在這裡不再對false的取值進行測試主要來測試true和extra的區別。我們再執行個體一的基礎上引入Dept(部門)類來進行測試:

部門實體類

部門表對應檔:

測試代碼:

lazy屬性值為true:

Lazy屬性值為extra:

有延時載入而引發的no Session問題

當我們在編寫基於分層的B/s程式時常常會因為Session提前關閉而資料沒有載入完成而引發no Session的異常:

該異常引發的原因時同城操作資料的代碼編寫在DAO層和Biz層但是這兩層並不負責資料的展示而我們在jsp頁面中對資料進行展示時Session早已關閉並且有與消極式載入的關係資料並沒有載入到對象中,當jsp頁面去訪問對象屬性時Hibernate嘗試使用Session對象去和資料庫互動時發現並沒有可用的Session對象從而引發該異常。

對於此類問題同城的做法是利用過濾器(Filter)將Session對象存放在展示層。如下代碼所示建立過濾器:

然後在web.xml檔案中配置過濾器:

經過以上操作就可以解決no Session的問題。當然還有其他的解決方案,但這種方案是使用最多的也是較為完善的解決方案,

緩衝      

緩衝的定義

緩衝是為了減少應用程式和資料庫互動次數而將一些修改頻率較低、查詢頻繁的非關鍵性資料單獨開闢一塊空間存放起來的一塊空間!是以一定範圍內的空間換取使用者從資料庫查詢資料的速度和效能的一種解決方案!

通常緩衝分為以下幾類:

內部緩衝、二級緩衝、查詢快取以及第三方緩衝實現。

內部緩衝

在Hibernate中內部緩衝又稱為一級緩衝和事務級緩衝由Hibernate自動維護不可卸載。其生命週期和Session對象的生命週期相同,當Session關閉時該緩衝也會被自動回收。如下所示:

其運行結果如下:

從結果中我們可以發現兩次查詢資料庫時Hibernate值是產生了一條sql語句也就是說只有第一次查詢時和資料庫進行了互動,並將查詢出的對象放入了內部緩衝當第二次查詢時Hibernate發現內部緩衝中已經存在該對象則直接將該對象返回不在和資料庫進行互動,並且這兩次查詢的對象的記憶體位址是完全相同的,由此可以得出內部緩衝中緩衝的是對象的記憶體位址的引用而不是對象的各個屬性值!

二級緩衝

二級緩衝是可配置的外掛程式,是進程或叢集範圍內的緩衝,可以被所有的Session共用

二級緩衝的配置

在Hibernate中配置二級緩衝的外掛程式有很多下面使用EHCache外掛程式為例來配置二級緩衝。

1.引入如下jar包。

      ehcache-1.2.3.jar  核心庫

      backport-util-concurrent.jar

      commons-logging.jar

   2.配置Hibernate.cfg.xml開啟二級緩衝

<propertyname="hibernate.cache.use_second_level_cache">true</property>

3.配置二級緩衝的供應商

<property name="hibernate.cache.provider_class">org.hibernate.cache.EhCacheProvider</property>

註:property元素必須在mapping元素之上

4.配置可進入二級緩衝的類

<class-cache usage="read-write" class="cn.happy.entity.Emp"/>

5.在Classpath目錄下引入ehcache.xml檔案

經過以上5個步驟就可以將Dept對象放入二級緩衝了,下面編寫測試案例

測試結果:

從結果中我們可以開出第二次查詢部門時並沒有產生產生sql語句但是我們兩次列印出的對象卻不是同一個記憶體位址,這是因為二級緩衝中存放的並不是對象象的記憶體位址的引用而是對象的散裝屬性(可以看成是對象的各個屬性值)所以我們訪問二級換存時需要將這些散裝屬性重新再記憶體中拼裝成一個完整的對象。如果你仔細看的話你會發現當我們第二次去訪問員工集合時Hibernate還是會產生sql語句。那是因為以上二級緩衝的配置是針對類針對類層級的。如果想要將員工集合也進行緩衝的話就需要在hibernate.cfg.xml設定檔中加入集合緩衝的配置,內容如下:

註:該配置必須是mapping元素的下一個元素

這時我們再來運行測試用來結果如下:

看到結果後你可能會大吃一驚,我沒配集合緩衝時他只產生一條sql語句我陪完之後怎麼變成兩條sql了?但是你注意看著兩條sql語句是一模一樣的,都是根據員工對象的OID來進行查詢的!那麼員工對象的OID是從哪來的呢?沒錯,就是從二級緩衝中擷取的至於為什麼只是緩衝員工對象的OID而沒有緩衝其他屬性值,是因為員工對象沒有配置二級緩衝其不能進入二級緩衝也就沒有辦法從二級緩衝中拿到它的其他屬性值,我們對員工對象配置二級緩衝後再來進行測試。

其結果如下:

查詢快取 查詢快取的配置

在hibernate.cfg.xml設定檔中加入上述元素開啟二級緩衝。並且在使用Query進行緩衝時必須在擷取集合前調用query1.setCacheable(true);

下面編寫測試案例進行測試:

測試結果:

注意:

1.查詢快取是基於二級緩衝的在配置查詢快取時必須配置二級緩衝否則將拋出如下異常

2.在查詢快取中儲存的是對象記憶體位址的引用而不是對象的散裝屬性。

3.查詢快取是根據兩次HQL查詢語句經Hibernate內部轉化後產生的sql語句是否一樣留在決定是否和資料庫進行互動而不是根據其對象的OID如下面的測試案例雖然查詢的是OID相同的對象但還是會和資料庫進行互動!

測試結果:

消極式載入和緩衝遺留問題 Lazy屬性和fetch屬性連用

在一對多或者多對多檢索策略由lazy和fetch共同確定,Lazy:決定關聯對象初始化時機,Fetch:決定SQL語句構建形式。Fetch屬性的取值為:Join:迫切左外串連、Select:多條簡單SQL(預設值)、Subselect:子查詢。當Fetch屬性取值為join是將忽略lazy屬性採用立即載入策略。例如插敘編號為5的部門,即便將部門對應檔中映射員工集合的set元素中的lazy屬性設定為true或extra其還是會立即載入出該部門下的所有員工的集合,其Hibernate內部產生的sql語句如下:

Query介面的list方法和iterate方法的區別

1.返回的類型不一樣,list返回List,iterate返回Iterator,
2.擷取資料的方式不一樣,list會直接查資料庫,iterate會先到資料庫中把id都取出來,然後真正要遍曆某個對象的時候先到緩衝中找,如果找不到,以id為條件再發一條sql到資料庫,這樣如果緩衝中沒有資料,則查詢資料庫的次數為n+1。
3.iterate會查詢2級緩衝,list 只會緩衝,但不會使用緩衝(除非結合查詢快取)。
4.list中返回的List中每個對象都是原本的對象,iterate中返回的對象是代理對象

緩衝的內部儲存實現

1.在緩衝中都是以map集合的形式對象資料

2.一級緩衝和二級緩衝中都是以對象的OID作為map結合的key值而查詢快取是以Hibernate內部產生的sql語句作為key值

3.一級緩衝和查詢快取中map集合的value值存放的是記憶體對象的引用,而二級緩衝中存放的是對象的散裝屬性。

4.當應用程式需要從資料庫擷取資料時Hibernate會以此檢索一級緩衝或查詢快取>二級緩衝或查詢快取中是否有合格資料如果有則直接返回不再進行和資料庫的互動,反之則產生sql語句去和資料庫進行互動擷取相應的資料再進行返回並依次將給資料放入一級緩衝>二級緩衝或查詢快取

將二級緩衝儲存到硬碟

在classpath目錄下的ehcache.xml設定檔中的diskStore元素的path屬性值設定為想要儲存的路徑並且將cache元素中的maxElementsInMemory屬性值設定為0。

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.