欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java?hibernate延遲加載get和load的區(qū)別

 更新時(shí)間:2022年09月23日 10:31:51   作者:文渡???????  
這篇文章主要介紹了Java?hibernate延遲加載get和load的區(qū)別,文章圍繞主題展開詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,需要的小伙伴可以參考一下

前言

在hibernate中我們知道如果要從數(shù)據(jù)庫中得到一個(gè)對象,通常有兩種方式,一種是通過session.get()方法,另一種就是通過session.load()方法,然后其實(shí)這兩種方法在獲得一個(gè)實(shí)體對象時(shí)是有區(qū)別的,在查詢性能上兩者是不同的。

一.load加載方式

當(dāng)使用load方法來得到一個(gè)對象時(shí),此時(shí)hibernate會使用延遲加載的機(jī)制來加載這個(gè)對象,即:當(dāng)我們使用session.load()方法來加載一個(gè)對象時(shí),此時(shí)并不會發(fā)出sql語句,當(dāng)前得到的這個(gè)對象其實(shí)是一個(gè)代理對象,這個(gè)代理對象只保存了實(shí)體對象的id值,只有當(dāng)我們要使用這個(gè)對象,得到其它屬性時(shí),這個(gè)時(shí)候才會發(fā)出sql語句,從數(shù)據(jù)庫中去查詢我們的對象。

session = HibernateUtil.openSession(); /* * 通過load的方式加載對象時(shí),會使用延遲加載機(jī)制,此時(shí)并不會發(fā)出sql語句,只有當(dāng)我們需要使用的時(shí)候才會從數(shù)據(jù)庫中去查詢 */ User user = (User)session.load(User.class, 2);

我們看到,如果我們僅僅是通過load來加載我們的User對象,此時(shí)從控制臺我們會發(fā)現(xiàn)并不會從數(shù)據(jù)庫中查詢出該對象,即并不會發(fā)出sql語句,但如果我們要使用該對象時(shí):

session = HibernateUtil.openSession();
      User user = (User)session.load(User.class, 2);
      System.out.println(user);

此時(shí)我們看到控制臺會發(fā)出了sql查詢語句,會將該對象從數(shù)據(jù)庫中查詢出來:

Hibernate: select user0_.id as id0_0_, user0_.username as username0_0_, user0_.password as password0_0_, user0_.born as born0_0_ from user user0_ where user0_.id=? User [id=2, username=aaa, password=111, born=2013-10-16 00:14:24.0]

這個(gè)時(shí)候我們可能會想,那么既然調(diào)用load方法時(shí),并不會發(fā)出sql語句去從數(shù)據(jù)庫中查出該對象,那么這個(gè)User對象到底是個(gè)什么對象呢?

其實(shí)這個(gè)User對象是我們的一個(gè)代理對象,這個(gè)代理對象僅僅保存了id這個(gè)屬性:

session = HibernateUtil.openSession(); /* * 通過load的方式加載對象時(shí),會使用延遲加載機(jī)制,此時(shí)得到的User對象其實(shí)是一個(gè)
             * 代理對象,該代理對象里面僅僅只有id這個(gè)屬性 */ User user = (User)session.load(User.class, 2);
            System.out.println(user.getId());

      console:  2 

我們看到,如果我們只打印出這個(gè)user對象的id值時(shí),此時(shí)控制臺會打印出該id值,但是同樣不會發(fā)出sql語句去從數(shù)據(jù)庫中去查詢。這就印證了我們的這個(gè)user對象僅僅是一個(gè)保存了id的代理對象,但如果我需要打印出user對象的其他屬性值時(shí),這個(gè)時(shí)候會不會發(fā)出sql語句呢?答案是肯定的:

 session = HibernateUtil.openSession(); /* * 通過load的方式加載對象時(shí),會使用延遲加載機(jī)制,此時(shí)得到的User對象其實(shí)是一個(gè)
             * 代理對象,該代理對象里面僅僅只有id這個(gè)屬性 */ User user = (User)session.load(User.class, 2);
            System.out.println(user.getId()); // 如果此時(shí)要得到user其他屬性,則會從數(shù)據(jù)庫中查詢
            System.out.println(user.getUsername());   

此時(shí)我們看控制臺的輸出:

2 Hibernate: select user0_.id as id0_0_, user0_.username as username0_0_, user0_.password as password0_0_, user0_.born as born0_0_ from user user0_ where user0_.id=? aaa

相信通過上述的幾個(gè)例子,大家應(yīng)該很好的了解了load的這種加載對象的方式了吧。

二、get加載方式

相對于load的延遲加載方式,get就直接的多,當(dāng)我們使用session.get()方法來得到一個(gè)對象時(shí),不管我們使不使用這個(gè)對象,此時(shí)都會發(fā)出sql語句去從數(shù)據(jù)庫中查詢出來:

session = HibernateUtil.openSession(); /* * 通過get方法來加載對象時(shí),不管使不使用該對象,都會發(fā)出sql語句,從數(shù)據(jù)庫中查詢 */ User user = (User)session.get(User.class, 2);

此時(shí)我們通過get方式來得到user對象,但是我們并沒有使用它,但是我們發(fā)現(xiàn)控制臺會輸出sql的查詢語句:

Hibernate: select user0_.id as id0_0_, user0_.username as username0_0_, user0_.password as password0_0_, user0_.born as born0_0_ from user user0_ where user0_.id=?

因此我們可以看到,使用load的加載方式比get的加載方式性能要好一些,因?yàn)閘oad加載時(shí),得到的只是一個(gè)代理對象,當(dāng)真正需要使用這個(gè)對象時(shí)再去從數(shù)據(jù)庫中查詢。

三、使用get和load時(shí)的一些小問題

當(dāng)了解了load和get的加載機(jī)制以后,我們此時(shí)來看看這兩種方式會出現(xiàn)的一些小問題:

①如果使用get方式來加載對象,當(dāng)我們試圖得到一個(gè)id不存在的對象時(shí),此時(shí)會報(bào)NullPointException的異常

session = HibernateUtil.openSession(); /* * 當(dāng)通過get方式試圖得到一個(gè)id不存在的user對象時(shí),此時(shí)會報(bào)NullPointException異常 */ User user = (User)session.get(User.class, 20);
            System.out.println(user.getUsername());

此時(shí)我們看控制臺的輸出信息,會報(bào)空指針的異常:

Hibernate: select user0_.id as id0_0_, user0_.username as username0_0_, user0_.password as password0_0_, user0_.born as born0_0_ from user user0_ where user0_.id=? java.lang.NullPointerException  .........

這是因?yàn)橥ㄟ^get方式我們會去數(shù)據(jù)庫中查詢出該對象,但是這個(gè)id值不存在,所以此時(shí)user對象是null,所以就會報(bào)NullPointException的異常了。

②如果使用load方式來加載對象,當(dāng)我們試圖得到一個(gè)id不存在的對象時(shí),此時(shí)會報(bào)ObjectNotFoundException異常:

session = HibernateUtil.openSession(); /* * 當(dāng)通過get方式試圖得到一個(gè)id不存在的user對象時(shí),此時(shí)會報(bào)ObjectNotFoundException異常 */ User user = (User)session.load(User.class, 20);
            System.out.println(user.getId());
            System.out.println(user.getUsername());

我們看看控制臺的輸出:

20 Hibernate: select user0_.id as id0_0_, user0_.username as username0_0_, user0_.password as password0_0_, user0_.born as born0_0_ from user user0_ where user0_.id=? org.hibernate.ObjectNotFoundException: No row with the given identifier exists: [com.xiaoluo.bean.User#20]......

為什么使用load的方式和get的方式來得到一個(gè)不存在的對象報(bào)的異常不同呢??其原因還是因?yàn)閘oad的延遲加載機(jī)制,使用load時(shí),此時(shí)的user對象是一個(gè)代理對象,僅僅保存了當(dāng)前的這個(gè)id值,當(dāng)我們試圖得到該對象的username屬性時(shí),這個(gè)屬性其實(shí)是不存在的,所以就會報(bào)出ObjectNotFoundException這個(gè)異常了。

③org.hibernate.LazyInitializationException異常

接下來我們再來看一個(gè)例子:

public class UserDAO
{ public User loadUser(int id)
    {
        Session session = null;
        Transaction tx = null;
        User user =  null; try {
            session = HibernateUtil.openSession();
            tx = session.beginTransaction();
            user = (User)session.load(User.class, 1);
            tx.commit();
        } catch (Exception e)
        {
            e.printStackTrace();
            tx.rollback();
        } finally {
            HibernateUtil.close(session);
        } return user;
    }
} 
@Test public void testLazy06()
    {
        UserDAO userDAO = new UserDAO();
        User user = userDAO.loadUser(2);
        System.out.println(user);
    }

模擬了一個(gè)UserDAO這樣的對象,然后我們在測試用例里面來通過load加載一個(gè)對象,此時(shí)我們發(fā)現(xiàn)控制臺會報(bào)LazyInitializationException異常

org.hibernate.LazyInitializationException: could not initialize proxy - no Session  .............

這個(gè)異常是什么原因呢??還是因?yàn)閘oad的延遲加載機(jī)制,當(dāng)我們通過load()方法來加載一個(gè)對象時(shí),此時(shí)并沒有發(fā)出sql語句去從數(shù)據(jù)庫中查詢出該對象,當(dāng)前這個(gè)對象僅僅是一個(gè)只有id的代理對象,我們還并沒有使用該對象,但是此時(shí)我們的session已經(jīng)關(guān)閉了,所以當(dāng)我們在測試用例中使用該對象時(shí)就會報(bào)LazyInitializationException這個(gè)異常了。

所以以后我們只要看到控制臺報(bào)LazyInitializationException這種異常,就知道是使用了load的方式延遲加載一個(gè)對象了,解決這個(gè)的方法有兩種,一種是將load改成get的方式來得到該對象,另一種是在表示層來開啟我們的session和關(guān)閉session。

至此,hibernate的兩種加載方式get和load已經(jīng)分析完畢!??!

到此這篇關(guān)于Java hibernate延遲加載get和load的區(qū)別的文章就介紹到這了,更多相關(guān)Java hibernate 內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • SpringBoot四種讀取properties文件的方式(小結(jié))

    SpringBoot四種讀取properties文件的方式(小結(jié))

    這篇文章主要介紹了SpringBoot四種讀取properties文件的方式,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-05-05
  • 計(jì)算Java數(shù)組長度函數(shù)的方法以及代碼分析

    計(jì)算Java數(shù)組長度函數(shù)的方法以及代碼分析

    在本篇內(nèi)容里,小編給大家整理了關(guān)于計(jì)算Java數(shù)組長度函數(shù)的方法以及代碼分析內(nèi)容,有興趣的朋友么可以學(xué)習(xí)參考下。
    2022-11-11
  • Java由淺入深講解繼承下

    Java由淺入深講解繼承下

    繼承就是可以直接使用前輩的屬性和方法。自然界如果沒有繼承,那一切都是處于混沌狀態(tài)。多態(tài)是同一個(gè)行為具有多個(gè)不同表現(xiàn)形式或形態(tài)的能力。多態(tài)就是同一個(gè)接口,使用不同的實(shí)例而執(zhí)行不同操作
    2022-04-04
  • SpringBoot使用Nacos配置中心的實(shí)現(xiàn)

    SpringBoot使用Nacos配置中心的實(shí)現(xiàn)

    這篇文章主要介紹了SpringBoot使用Nacos配置中心的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-12-12
  • java代碼實(shí)現(xiàn)MD5加密及驗(yàn)證過程詳解

    java代碼實(shí)現(xiàn)MD5加密及驗(yàn)證過程詳解

    這篇文章主要介紹了java代碼實(shí)現(xiàn)MD5加密及驗(yàn)證過程詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2019-10-10
  • IntelliJ IDEA 2020.2 配置大全詳細(xì)圖文教程(更新中)

    IntelliJ IDEA 2020.2 配置大全詳細(xì)圖文教程(更新中)

    這篇文章主要介紹了IntelliJ IDEA 2020.2 配置大全(更新中),本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-08-08
  • java實(shí)現(xiàn)電腦端掃描二維碼

    java實(shí)現(xiàn)電腦端掃描二維碼

    這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)電腦端掃描二維碼,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2018-10-10
  • Java嵌套類和內(nèi)部類詳解

    Java嵌套類和內(nèi)部類詳解

    這篇文章主要介紹了Java嵌套類和內(nèi)部類詳解,本文講解了什么是嵌套類及內(nèi)部類、靜態(tài)嵌套類、在外部類中定義內(nèi)部類、在方法中定義內(nèi)部類、匿名內(nèi)部類等內(nèi)容,需要的朋友可以參考下
    2015-04-04
  • java中單例模式講解

    java中單例模式講解

    這篇文章主要介紹了java中單例模式,本文通過簡單的案例,講解了該模式在java中的使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下
    2021-08-08
  • MybatisX中xml映射文件中命名空間爆紅的解決

    MybatisX中xml映射文件中命名空間爆紅的解決

    本文主要介紹了MybatisX中xml映射文件中命名空間爆紅的解決,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-06-06

最新評論