Hibernate中Session.get()方法和load()方法的詳細(xì)比較
一、get方法和load方法的簡(jiǎn)易理解
(1)get()方法直接返回實(shí)體類,如果查不到數(shù)據(jù)則返回null。load()會(huì)返回一個(gè)實(shí)體代理對(duì)象(當(dāng)前這個(gè)對(duì)象可以自動(dòng)轉(zhuǎn)化為實(shí)體對(duì)象),但當(dāng)代理對(duì)象被調(diào)用時(shí),如果沒有數(shù)據(jù)不存在,就會(huì)拋出個(gè)org.hibernate.ObjectNotFoundException異常
(2)load先到緩存(session緩存/二級(jí)緩存)中去查,如果沒有則返回一個(gè)代理對(duì)象(不馬上到DB中去找),等后面使用這個(gè)代理對(duì)象操作的時(shí)候,才到DB中查詢,這就是我們常說的 load在默認(rèn)情況下支持延遲加載(lazy)
(3) get先到緩存(session緩存/二級(jí)緩存)中去查,如果沒有就到DB中去查(即馬上發(fā)出sql)??傊?,如果你確定DB中有這個(gè)對(duì)象就用load(),不確定就用get()(這樣效率高)
(4)get和load查詢數(shù)據(jù)庫(kù)
1. 如果查詢不到數(shù)據(jù),get 會(huì)返回 null,但是不會(huì)報(bào)錯(cuò), load 如果查詢不到數(shù)據(jù),則報(bào)錯(cuò)ObjectNotFoundException
2. 使用get 去查詢數(shù)據(jù),(先到一級(jí)/二級(jí))會(huì)立即向db發(fā)出查詢請(qǐng)求(select ...), 如果你使用的是 load查詢數(shù)據(jù),(先到一級(jí)、二級(jí)))即使查詢到對(duì)象,返回的是一個(gè)代理對(duì)象,如果后面沒有使用查詢結(jié)果,它不會(huì)真的向數(shù)據(jù)庫(kù)發(fā)select ,當(dāng)程序員使用查詢結(jié)果的時(shí)候才真的發(fā)出select ,這個(gè)現(xiàn)象我們稱為懶加載(lazy)
3. 通過修改配置文件(*.hbm.xml文件),我們可以取消懶加載
<class name="Employee" lazy="false" table="employee">
二、結(jié)合緩存技術(shù)比較get和load區(qū)別

注:一級(jí)緩存是必須有的,二級(jí)緩存是配置的可選(這里解釋的時(shí)候默認(rèn)有二級(jí)緩存)
1.get在查詢的時(shí)候首先到一級(jí)緩存(session級(jí)緩存)中查找,沒有的話進(jìn)入二級(jí)緩存(介于內(nèi)存與硬盤之間) 如果二級(jí)緩存還沒有則直接進(jìn)入數(shù)據(jù)庫(kù)中查詢!查詢到并將查詢的結(jié)果放入二級(jí)緩存!查詢不到不會(huì)報(bào)錯(cuò) 如果查詢同一條數(shù)據(jù),get會(huì)出現(xiàn)在第一次查詢的時(shí)候會(huì)出查詢數(shù)據(jù)庫(kù) 但是第二次的時(shí)候會(huì)直接沿著一級(jí)緩存、二級(jí)緩存查找可以找到!
2.load在查詢的時(shí)候也是先到一級(jí)緩存中查找,如果找到則不會(huì)查詢數(shù)據(jù)庫(kù),如果找不到則依次到二級(jí)緩存查找,如果二級(jí)緩存中沒有查找的數(shù)據(jù),get方法會(huì)根據(jù)是否要立即利用查詢的結(jié)果,如果不利用的話則停止查找;如果要立即使用的話則會(huì)進(jìn)入數(shù)據(jù)庫(kù)中查找;查詢到并將查詢的結(jié)果放入二級(jí)緩存!查詢不到則會(huì)報(bào)錯(cuò)!
三、深入?yún)^(qū)別get方法和load方法
(1)hibernate中g(shù)et方法和load方法的根本區(qū)別在于:
1.如果你使用load方法,hibernate認(rèn)為該id對(duì)應(yīng)的對(duì)象(數(shù)據(jù)庫(kù)記錄)在數(shù)據(jù)庫(kù)中是一定存在的,所以它可以放心的使用,它可以放心的使用代理來延遲加載該對(duì)象。在用到對(duì)象中的其他屬性數(shù)據(jù)時(shí)才查詢數(shù)據(jù)庫(kù),但是萬一數(shù)據(jù)庫(kù)中不存在該記錄,那沒辦法,只能拋異常,所說的load方法拋異常是指在使用該對(duì)象的數(shù)據(jù)時(shí),數(shù)據(jù)庫(kù)中不存在該數(shù)據(jù)時(shí)拋異常,而不是在創(chuàng)建這個(gè)對(duì)象時(shí)。
2.由于 session中的緩存對(duì)于hibernate來說是個(gè)相當(dāng)廉價(jià)的資源,所以在load時(shí)會(huì)先查一下session緩存看看該id對(duì)應(yīng)的對(duì)象是否存在,不存在則創(chuàng)建代理。所以如果你知道該id在數(shù)據(jù)庫(kù)中一定有對(duì)應(yīng)記錄存在就可以使用load方法來實(shí)現(xiàn)延遲加載。
3.對(duì)于get方法,hibernate會(huì)確認(rèn)一下該id對(duì)應(yīng)的數(shù)據(jù)是否存在,首先在session緩存中查找,然后在二級(jí)緩存中查找,還沒有就查數(shù)據(jù)庫(kù),數(shù)據(jù)庫(kù)中沒有就返回null。
4.get方法如果在session緩存中找到了該id 對(duì)應(yīng)的對(duì)象,如果剛好該對(duì)象前面是被代理過的,如被load方法使用過,或者被其他關(guān)聯(lián)對(duì)象延遲加載過,那么返回的還是原先的代理對(duì)象,而不是實(shí)體類對(duì)象,如果該代理對(duì)象還沒有加載實(shí)體數(shù)據(jù)(就是id以外的其他屬性數(shù)據(jù)),那么它會(huì)查詢二級(jí)緩存或者數(shù)據(jù)庫(kù)來加載數(shù)據(jù),但是返回的還是代理對(duì)象,只不過已經(jīng)加載了實(shí)體數(shù)據(jù)。
5.get方法首先查詢session緩存,沒有的話查詢二級(jí)緩存,最后查詢數(shù)據(jù)庫(kù);反而load方法創(chuàng)建時(shí)首先查詢session緩存,沒有就創(chuàng)建代理,實(shí)際使用數(shù)據(jù)時(shí)才查詢二級(jí)緩存和數(shù)據(jù)庫(kù)。
(2)總之對(duì)于get和load的根本區(qū)別,一句話,hibernate對(duì)于load方法認(rèn)為該數(shù)據(jù)在數(shù)據(jù)庫(kù)中一定存在,可以放心的使用代理來延遲加載,如果在使用過程中發(fā)現(xiàn)了問題,只能拋異常;而對(duì)于get方法,hibernate一定要獲取到真實(shí)的數(shù)據(jù),否則返回null。
四、load和get方法的使用
/**
* 比較load方法和get方法的區(qū)別
**/
public static void CompareLoadAndGet(){
Configuration configuration = new Configuration().configure();
SessionFactory sessionFactory = configuration.buildSessionFactory();
Session session = sessionFactory.openSession();
Transaction ts = null;
Transaction ts1 = null;
try {
/**
* 使用load方法
**/
ts = session.beginTransaction();
Employee emp= (Employee) session.load(Employee.class, 9);
System.out.println("load方法獲取的員工姓名:"+emp.getName());
ts.commit();
/**
* 使用get方法
**/
ts1 = session.beginTransaction();
Employee emp1= (Employee) session.get(Employee.class, 9);
System.out.println("get方法獲取的員工姓名:"+emp1.getName());
ts1.commit();
}
// catch (Exception e) {
// if(ts!=null){
// ts.rollback();
// }
// }
finally{
if(session!=null&&session.isOpen()){
session.close();
}
}
}
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)腳本之家的支持。如果你想了解更多相關(guān)內(nèi)容請(qǐng)查看下面相關(guān)鏈接
相關(guān)文章
Spring Boot 實(shí)現(xiàn)程序的優(yōu)雅退出(詳細(xì)步驟)
Spring Boot 為我們提供了優(yōu)雅退出的功能,使應(yīng)用程序能夠在關(guān)閉時(shí)正常處理完所有當(dāng)前請(qǐng)求,避免請(qǐng)求被中斷導(dǎo)致數(shù)據(jù)丟失或不一致等問題,本文將全面介紹如何在 Spring Boot 應(yīng)用程序中實(shí)現(xiàn)優(yōu)雅退出,感興趣的朋友跟隨小編一起看看吧2024-03-03
JMeter中的后端監(jiān)聽器的實(shí)現(xiàn)
本文主要介紹了JMeter中的后端監(jiān)聽器的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-09-09
mybatis 一對(duì)多嵌套查詢的實(shí)現(xiàn)
本文主要介紹了mybatis 一對(duì)多嵌套查詢的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2024-07-07
Spring security用戶URL權(quán)限FilterSecurityInterceptor使用解析
這篇文章主要介紹了Spring security用戶URL權(quán)限FilterSecurityInterceptor使用解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-12-12
java多線程之Future和FutureTask使用實(shí)例
這篇文章主要介紹了java多線程之Future和FutureTask使用實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-09-09
源碼解析springbatch的job運(yùn)行機(jī)制
這篇文章主要介紹了springbatch的job是如何運(yùn)行的,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-08-08
關(guān)于java.lang.NumberFormatException: null的問題及解決
這篇文章主要介紹了關(guān)于java.lang.NumberFormatException: null的問題及解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-09-09

