hibernate 中 get、load 的 區別

轉自:http://blog.163.com/love-love-l/blog/static/210783042009618919985/#comment=fks_083070085095080069081094083095086087081068092085086071

從網上抄了兩片。感覺不錯,呵呵。

第一篇

這次我們來談一下Hibernate3.2 Session加載數據時get和load方法的區別(Hibernate 3以後的版本就用get()方法取代find()這個方法了),其實這個在網上有很多的論述,可大多語焉不詳或經不起實踐的推敲,讓很多初學者學的滿腹疑 竇,現在我給大家講解一下:

1. 對於Hibernate get方法,Hibernate會確認一下該id對應的數據是否存在,首先在session緩存中查找,然後在二級緩存中查找,還沒有就查詢數據庫,數據 庫中沒有就返回null。這個相對比較簡單,也沒有太大的爭議。主要要說明的一點就是在這個版本中get方法也會查找二級緩存!

2. Hibernate load方法加載實體對象的時候,根據映射文件上類級別的lazy屬性的配置(默認爲true),分情況討論:

(1)若爲true,則首先在Session緩存中查找,看看該id對應的對象是否存在,不存在則使用延遲加載,返回實體的代理類對象(該代理類爲 實體類的子類,由CGLIB動態生成)。等到具體使用該對象(除獲取OID以外)的時候,再查詢二級緩存和數據庫,若仍沒發現符合條件的記錄,則會拋出一 個ObjectNotFoundException。

(2)若爲false,就跟Hibernate get方法查找順序一樣,只是最終若沒發現符合條件的記錄,則會拋出一個ObjectNotFoundException。

這裏get和load有兩個重要區別:

如果未能發現符合條件的記錄,Hibernate get方法返回null,而load方法會拋出一個ObjectNotFoundException。 
load 方法可返回沒有加載實體數據的代理類實例,而get方法永遠返回有實體數據的對象。(對於load和get方法返回類型:好多書中都說:“get方法永遠 只返回實體類”,實際上並不正確,get方法如果在session緩存中找到了該id對應的對象,如果剛好該對象前面是被代理過的,如被load方法使用 過,或者被其他關聯對象延遲加載過,那麼返回的還是原先的代理對象,而不是實體類對象,如果該代理對象還沒有加載實體數據(就是id以外的其他屬性數 據),那麼它會查詢二級緩存或者數據庫來加載數據,但是返回的還是代理對象,只不過已經加載了實體數據。) 
總之對於get和load的根本區別,一句話,hibernate對於load方法認爲該數據在數據庫中一定存在,可以放心的使用代理來延遲加載,如果在使用過程中發現了問題,只能拋異常;而對於get方法,hibernate一定要獲取到真實的數據,否則返回null。

最後,我們分析下爲什麼網路上那麼多關於二者區別的文章不太準確!首先可能是版本問題,Hibernate版本不同,運行機制不太一樣;其次就是很 多朋友只是把自己經驗所得與人分享,並沒有經過全方位代碼的檢測;最後就是有些技術牛人,表達比較隨意。所以我希望大家以後學習知識的時候不要盲從他人迷 信權威,一定要綜合多方資料,比較和整理,再經過自己實踐檢驗,這樣得到的知識才是真實有效的。

 

第二篇

  1. Users user = (Users)session.load(Users. class , userId);    

get加載方法:

Java代碼 
  1. Users user = (Users)session.get(Users. class , userId);  

 

兩加載方法區別:

區別1:如果數據庫中,沒有userId的對象。如果通過get方法加載,則返回的是一個null;如果通過load加載,則返回一個代理對象,如果後面代碼如果調用user對象的某個屬性(比如user.getPassword())會拋出異常:org.hibernate.ObjectNotFoundException;

區別2:load支持延遲加載,get不支持延遲加載。

也就是說:
Java代碼 
  1. Users user = (Users)session.load(Users. class , userId);  

這句代碼不會去執行數據庫查詢,只有用到user時纔會去執行數據庫查詢。

而:
Java代碼 
  1. Users user = (Users)session.get(Users. class , userId);  

則立即去執行數據庫查詢。 所以Users user = (Users)session.load(Users.class, userId);不會執行任何sql。

注意:
Java代碼 
  1. Users user = (Users)session.load(Users. class , userId);  
  2. System.out.println(user.getId());  

上面這2句代碼,不會去執行數據庫操作。因爲load後會在hibernate 的一級緩存裏存放一個map對象,該map的key就是userId的值,但是當你getId()時,它會去一級緩存裏拿map的key值,而不去執行數據庫查詢。所以不會報任何錯。不會執行任何數據庫操作。

發佈了11 篇原創文章 · 獲贊 1 · 訪問量 10萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章