亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 開發 > 綜合 > 正文

hibernate延遲加載(get和load的區別)

2024-07-21 02:51:21
字體:
來源:轉載
供稿:網友

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

一.load加載方式

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

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

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

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

此時我們看到控制臺會發出了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=?User [id=2, username=aaa, password=111, born=2013-10-16 00:14:24.0]

這個時候我們可能會想,那么既然調用load方法時,并不會發出sql語句去從數據庫中查出該對象,那么這個User對象到底是個什么對象呢?

其實這個User對象是我們的一個代理對象,這個代理對象僅僅保存了id這個屬性:

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

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

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

此時我們看控制臺的輸出:

2Hibernate: 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

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

二、get加載方式

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

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

此時我們通過get方式來得到user對象,但是我們并沒有使用它,但是我們發現控制臺會輸出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的加載方式性能要好一些,因為load加載時,得到的只是一個代理對象,當真正需要使用這個對象時再去從數據庫中查詢。

三、使用get和load時的一些小問題

當了解了load和get的加載機制以后,我們此時來看看這兩種方式會出現的一些小問題:

①如果使用get方式來加載對象,當我們試圖得到一個id不存在的對象時,此時會報NullPointException的異常

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

此時我們看控制臺的輸出信息,會報空指針的異常:

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  .........

這是因為通過get方式我們會去數據庫中查詢出該對象,但是這個id值不存在,所以此時user對象是null,所以就會報NullPointException的異常了。

②如果使用load方式來加載對象,當我們試圖得到一個id不存在的對象時,此時會報ObjectNotFoundException異常:

復制代碼
      session = HibernateUtil.openSession();            /*             * 當通過get方式試圖得到一個id不存在的user對象時,此時會報ObjectNotFoundException異常             */            User user = (User)session.load(User.class, 20);            System.out.println(user.getId());            System.out.println(user.getUsername());復制代碼

我們看看控制臺的輸出:

20Hibernate: 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的方式來得到一個不存在的對象報的異常不同呢??其原因還是因為load的延遲加載機制,使用load時,此時的user對象是一個代理對象,僅僅保存了當前的這個id值,當我們試圖得到該對象的username屬性時,這個屬性其實是不存在的,所以就會報出ObjectNotFoundException這個異常了。

③org.hibernate.LazyInitializationException異常

接下來我們再來看一個例子:

復制代碼
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);    }復制代碼

模擬了一個UserDAO這樣的對象,然后我們在測試用例里面來通過load加載一個對象,此時我們發現控制臺會報LazyInitializationException異常

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

這個異常是什么原因呢??還是因為load的延遲加載機制,當我們通過load()方法來加載一個對象時,此時并沒有發出sql語句去從數據庫中查詢出該對象,當前這個對象僅僅是一個只有id的代理對象,我們還并沒有使用該對象,但是此時我們的session已經關閉了,所以當我們在測試用例中使用該對象時就會報LazyInitializationException這個異常了。

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


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国产黑人绿帽在线第一区| 色琪琪综合男人的天堂aⅴ视频| 91夜夜未满十八勿入爽爽影院| 91九色国产社区在线观看| 一本一本久久a久久精品牛牛影视| 久久精品人人做人人爽| 亚洲第一福利网站| 欧美激情伊人电影| 欧美www视频在线观看| 一个人看的www欧美| www.精品av.com| 欧美性资源免费| 亚洲激情视频网| 欧美精品做受xxx性少妇| 欧洲亚洲免费在线| 精品动漫一区二区| 久久香蕉国产线看观看网| 国产成+人+综合+亚洲欧洲| 国产在线高清精品| 中文字幕久久精品| 91视频国产一区| 亚洲码在线观看| 视频在线一区二区| 91免费精品国偷自产在线| 亚洲free性xxxx护士hd| 91成人在线观看国产| 精品香蕉在线观看视频一| 欧美日韩国产麻豆| 久久精品这里热有精品| 亚洲xxx大片| 亚洲精品久久久一区二区三区| 亚洲片在线观看| 国产成人精品999| 欧美午夜女人视频在线| 日韩精品一二三四区| 午夜精品久久17c| 亚洲欧洲在线播放| 亚洲色图13p| 欧美精品午夜视频| 爽爽爽爽爽爽爽成人免费观看| 色午夜这里只有精品| 欧美裸体xxxxx| 日韩不卡中文字幕| 国产91成人在在线播放| 91精品国产乱码久久久久久蜜臀| 91在线中文字幕| 亚洲国产精品成人精品| 欧美日韩国产一区二区| 亚洲综合日韩中文字幕v在线| 97视频在线观看免费高清完整版在线观看| 精品中文字幕久久久久久| 久久人91精品久久久久久不卡| 亚洲国产美女久久久久| 欧美xxxx做受欧美.88| 成人午夜小视频| 国内精品久久久久影院 日本资源| 97精品国产97久久久久久| 国产成人aa精品一区在线播放| 久久亚洲精品一区二区| 日韩成人在线观看| 亚洲第一视频网站| 美日韩在线视频| 欧美在线不卡区| 91久久久久久久| 成人激情在线观看| 亚洲国产中文字幕在线观看| 日韩精品极品视频免费观看| 奇门遁甲1982国语版免费观看高清| 国产精品成人va在线观看| 亚洲三级黄色在线观看| 国产精品久久久久久一区二区| 奇米成人av国产一区二区三区| 一本色道久久88综合亚洲精品ⅰ| 97在线免费观看| 欧美亚洲一级片| 免费不卡欧美自拍视频| 国产亚洲精品美女久久久久| 91国产一区在线| 欧美孕妇与黑人孕交| 成人性生交大片免费看视频直播| 久久久久久久久中文字幕| 97热在线精品视频在线观看| 午夜精品国产精品大乳美女| 黑人狂躁日本妞一区二区三区| 91av在线免费观看视频| 欧美激情第三页| 在线中文字幕日韩| 国产精品99久久久久久www| 欧美巨猛xxxx猛交黑人97人| 国产亚洲精品久久久久久牛牛| 国产最新精品视频| 亚洲图片在线综合| 亚洲欧美综合区自拍另类| 中文字幕免费精品一区| 日韩的一区二区| 日韩视频免费看| 亚洲天堂色网站| 日韩av资源在线播放| 日韩电影免费观看中文字幕| 亚洲va电影大全| 在线看福利67194| 成人久久久久久久| 亚洲春色另类小说| 日韩av免费观影| 欧美大奶子在线| 色综合视频一区中文字幕| 久久精品在线视频| 国内精品久久久久久中文字幕| 国产精品欧美日韩| 92裸体在线视频网站| 欧美电影第一页| 国内精品久久久久影院优| 伊人久久久久久久久久久久久| 美女撒尿一区二区三区| 国产成人精品优优av| 色先锋资源久久综合5566| 亚洲成人网在线观看| 欧美激情一区二区三区久久久| 97在线看免费观看视频在线观看| 一本一道久久a久久精品逆3p| 2025国产精品视频| 成人免费视频网| 精品国产一区二区三区久久久| 久久艳片www.17c.com| 亚洲xxxx做受欧美| 国产91在线播放| 色播久久人人爽人人爽人人片视av| 伊人激情综合网| 91a在线视频| 色先锋资源久久综合5566| 亚洲精品久久久久中文字幕二区| 成人久久久久久| 亚洲影视九九影院在线观看| 日本aⅴ大伊香蕉精品视频| 国产精品久久久久久久久久| 性色av一区二区三区在线观看| 97视频在线观看视频免费视频| 在线播放日韩av| 国产精品久久久久久av下载红粉| 日韩av电影在线播放| 成人日韩av在线| 国产精品va在线播放| 清纯唯美亚洲激情| 国产香蕉97碰碰久久人人| 亚洲午夜国产成人av电影男同| 国产欧美日韩免费看aⅴ视频| 欧洲日本亚洲国产区| 亚洲美女激情视频| 欧美一区二区三区免费视| 欧美日韩国内自拍| 国产综合在线观看视频| 欧美性视频网站| 高清日韩电视剧大全免费播放在线观看| 日韩在线观看免费网站| 欧美激情精品久久久久久蜜臀| 欧美自拍视频在线| 久久精品国产视频| 亚洲电影在线看| 国产精品欧美亚洲777777| 97人人模人人爽人人喊中文字| 国内精品免费午夜毛片| 97在线视频免费播放| 欧美黑人巨大xxx极品| 最新中文字幕亚洲|