Hibernate中延遲載入和緩存

来源:http://www.cnblogs.com/wangzheand/archive/2016/08/17/5781348.html
-Advertisement-
Play Games

什麼是延遲載入? 延遲載入是指當應用程式想要從資料庫獲取對象時(在沒有設置lazy屬性值為false),Hibernate只是從資料庫獲取符合條件的對象的OId從而生成代理對象,並沒有載入出對象 訪問該對象的屬性時才會載入出相應的值。簡答來說就是儘可能的減少查詢的數據量。 如何配置延遲載入 在Hib... ...


什麼是延遲載入?

延遲載入是指當應用程式想要從資料庫獲取對象時(在沒有設置lazy屬性值為false),Hibernate只是從資料庫獲取符合條件的對象的OId從而生成代理對象,並沒有載入出對象

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

如何配置延遲載入

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

類Class標簽中的lazy:

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

wpsCE55.tmp

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

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

員工表表結構:

wpsCE65.tmp

員工實體類:

wpsCE76.tmp

員工表的hbm映射文件:

wpsCE77.tmp

測試代碼:

wpsCE78.tmp

測試結果:

wpsCE79.tmp

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

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

wpsCE7A.tmp

wpsCE7B.tmp

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

註意:我們再上面編寫測試用例時獲取員工對象是用的get方法wpsCE7C.tmp而我也在之前的博客中說過session對象的get方法不支持延時載入他會忽略掉類級別的lazy屬性!我們把get方法換成load方法再來測試。

wpsCE7D.tmp

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

多對一關聯中的lazy               

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

wpsCE7E.tmp

比如我們在實例一的基礎上添加lazy屬性值為false再來測試:

wpsCE7F.tmp

Set元素中的lazy屬性

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

wpsCE80.tmp

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

部門實體類

wpsCE81.tmp

部門表映射文件:

wpsCE82.tmp

測試代碼:

wpsCE83.tmp

lazy屬性值為true:

wpsCE93.tmp

Lazy屬性值為extra:

wpsCE94.tmp

wpsCE95.tmp

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

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

wpsCE96.tmp

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

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

wpsCE97.tmp

然後在web.xml文件中配置過濾器:

wpsCE98.tmp

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

緩存      

緩存的定義

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

通常緩存分為以下幾類:

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

內部緩存

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

wpsCE99.tmp

其運行結果如下:

wpsCE9A.tmp

從結果中我們可以發現兩次查詢資料庫時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文件

wpsCE9B.tmp

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

wpsCE9C.tmp

測試結果:

wpsCE9D.tmp

wpsCE9E.tmp

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

wpsCE9F.tmp

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

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

wpsCEA0.tmp

wpsCEA1.tmp

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

wpsCEB2.tmp

其結果如下:

wpsCEB3.tmp

查詢緩存

查詢緩存的配置

wpsCEB4.tmp

在hibernate.cfg.xml配置文件中加入上述元素開啟二級緩存。並且在使用Query進行緩存時必須在獲取集合前調用query1.setCacheable(true);

下麵編寫測試用例進行測試:

wpsCEB5.tmp

測試結果:

wpsCEB6.tmp

註意:

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

2.在查詢緩存中保存的是對象記憶體地址的引用而不是對象的散裝屬性。

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

wpsCEB8.tmp

測試結果:

wpsCEB9.tmp

延遲載入和緩存遺留問題

Lazy屬性和fetch屬性連用

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

wpsCEBA.tmp

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。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1.簡介 在我們數據分析的實際應用中,我們可能會花費大量的時間在數據清洗上,而如果使用 R 裡面自帶的一些函數(base 包的 transform 等),可能會覺得力不從心,或者不是很人性化。好在我們有其他選擇。這裡我們介紹 dplyr 包。 首先載入包: 單表操作函數(one table verb ...
  • 1、PDO:PDO 簡介 PDO(PHP Data Object)即PHP數據對象,其為PHP訪問不同的資料庫定義了一致的介面。PDO只是抽象層面的介面,本身並不能提供訪問資料庫的能力,要想使用PDO訪問資料庫,需要為其指定一個實現了PDO介面的資料庫驅動。其類似於JAVA中的JDBC規範。 實現了 ...
  • 提交給ACM與IEEE的論文中(如CIKM),除了有明確具體字體類型的Type1和TrueType字體外,經常會出現Type3字體。 Type 3 字體是矢量外框字體,很多網站所稱Type 3是點陣字體其實是錯誤的。不同於 Type 1,PostScript Type 3 字體不支援 Hinting ...
  • 一、生成器 通過列表生成式,我們可以直接創建一個列表。但是,受到記憶體限制,列表容量肯定是有限的。而且,創建一個包含100萬個元素的列表,不僅占用很大的存儲空間,如果我們僅僅需要訪問前面幾個元素,那後面絕大多數元素占用的空間都白白浪費了。所以,如果列表元素可以按照某種演算法推算出來,那我們是否可以在迴圈 ...
  • 最近新寫的程式要做beta測試,在做綠色版(免安裝版)時遇到了問題,vs2015做的項目本以為像之前的vs版本一樣把msvcrXXX.dll還有另外幾個運行時庫都放到exe旁邊即可,然並卵。。。,在win7的測試機上還會提示找不到這個dll:api-ms-win-crt-runtime-l1-1-0 ...
  • 這篇博客總結了1天,希望自己以後返回來看的時候理解更深刻,也希望可以起到幫助初學者的作用. 轉載請註明 出自 : "luogg的博客園" , 因為前不久偶然發現某網站直接複製粘貼我的博客,交談之後他們修改了出處. 封裝 一, 封裝的概念 1. 類可以看成一個封裝體,將對象的屬性和方法封裝到類中. 2 ...
  • 在開發過程當中需要用到配置信息,這些信息不能進行硬編碼,這時配置文件是一個比較好的方式,java提供了properties格式的文件,以鍵值對的方式保存信息,在讀取的時候通過鍵獲得鍵對應的值,spring提供了讀取properties文件的支持,下麵看具體的配置, 一、<context:proper ...
  • --> 斷點續傳: 就像迅雷下載文件一樣,停止下載或關閉程式,下次下載時是從上次下載的地方開始繼續進行,而不是重頭開始... --> RandomAccessFile --> pointer(文件指針) --> seek(移動文件指針) --> 斷點續傳 --> 通過複製來模擬簡單的斷點續傳... ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...