
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版本不同,運行機制不太一樣;其次就是很多朋友只是把自己經驗所得與人分享,并沒有經過全方位代碼的檢測;最后就是有些技術牛人,表達比較隨意。所以我希望大家以后學習知識的時候不要盲從他人迷信權威,一定要綜合多方資料,比較和整理,再經過自己實踐檢驗,這樣得到的知識才是真實有效的。
第二篇
Users user = (Users)session.load(Users.class, userId);????
Users user = (Users)session.load(Users.class, userId);? get加載方法:
Java代碼
Users user = (Users)session.get(Users.class, userId);??
Users user = (Users)session.get(Users.class, userId);兩加載方法區別:
區別1:如果數據庫中,沒有userId的對象。如果通過get方法加載,則返回的是一個null;如果通過load加載,則返回一個代理對象,如果后面代碼如果調用user對象的某個屬性(比如user.getPassword())會拋出異常:org.hibernate.ObjectNotFoundException;
區別2:load支持延遲加載,get不支持延遲加載。
也就是說: 而:
Java代碼
Users user = (Users)session.load(Users.class, userId);??
Users user = (Users)session.load(Users.class, userId);這句代碼不會去執行數據庫查詢,只有用到user時才會去執行數據庫查詢。
Java代碼
Users user = (Users)session.get(Users.class, userId);??
Users user = (Users)session.get(Users.class, userId);則立即去執行數據庫查詢。 所以Users user = (Users)session.load(Users.class, userId);不會執行任何sql。
注意:
Java代碼
Users user = (Users)session.load(Users.class, userId);??
System.out.println(user.getId());??
Users user = (Users)session.load(Users.class, userId);
System.out.println(user.getId());上面這2句代碼,不會去執行數據庫操作。因為load后會在hibernate的一級緩存里存放一個map對象,該map的key就是userId的值,但是當你getId()時,它會去一級緩存里拿map的key值,而不去執行數據庫查詢。所以不會報任何錯。不會執行任何數據庫操作。
本文來自CSDN博客,轉載請標明出處:http://blog.csdn.net/shiyuntian_wang/archive/2009/09/28/4604595.aspx
更多文章、技術交流、商務合作、聯系博主
微信掃碼或搜索:z360901061

微信掃一掃加我為好友
QQ號聯系: 360901061
您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。
【本文對您有幫助就好】元
