BlogJava 首頁 新隨筆 聯(lián)系 聚合 管理
            37 Posts :: 64 Stories :: 21 Comments :: 0 Trackbacks

          Hibernate 完全以面向?qū)ο蟮姆绞絹聿僮鲾?shù)據(jù)庫,當(dāng)程序里以面向?qū)ο蟮姆绞讲僮鞒志没瘜ο髸r(shí),將被自動(dòng)轉(zhuǎn)換為對數(shù)據(jù)庫的操作。例如調(diào)用Session的 delete()方法來刪除持久化對象,Hibernate將負(fù)責(zé)刪除對應(yīng)的數(shù)據(jù)記錄;當(dāng)執(zhí)行持久化對象的set方法時(shí),Hibernate將自動(dòng)轉(zhuǎn)換為對應(yīng)的update方法,修改數(shù)據(jù)庫的對應(yīng)記錄。

          問題是如果需要同時(shí)更新100?000條記錄,是不是要逐一加載100?000條記錄,然后依次調(diào)用set方法——這樣不僅繁瑣,數(shù)據(jù)訪問的性能也十分糟糕。對這種批量處理的場景,Hibernate提供了批量處理的解決方案,下面分別從批量插入、批量更新和批量刪除3個(gè)方面介紹如何面對這種批量處理的情形。

          4.2.1 批量插入

          如果需要將100?000條記錄插入數(shù)據(jù)庫,通常Hibernate可能會采用如下做法:

          1. Session session = sessionFactory.openSession();   
          2. Transaction tx = session.beginTransaction();   
          3. for ( int i=0; i<100000; i++ ) {   
          4.     User u = new User (.....);   
          5.     session.save(customer);   
          6. }   
          7.   
          8. tx.commit();   
          9. session.close();  
          Session session = sessionFactory.openSession();
          Transaction tx = session.beginTransaction();
          for ( int i=0; i<100000; i++ ) {
          User u = new User (.....);
          session.save(customer);
          }
          tx.commit();
          session.close();

          但隨著這個(gè)程序的運(yùn)行,總會在某個(gè)時(shí)候運(yùn)行失敗,并且拋出OutOfMemoryException(內(nèi)存溢出異常)。這是因?yàn)镠ibernate的Session持有一個(gè)必選的一級緩存,所有的User實(shí)例都將在Session級別的緩存區(qū)進(jìn)行了緩存的緣故。

          為了解決這個(gè)問題,有個(gè)非常簡單的思路:定時(shí)將Session緩存的數(shù)據(jù)刷新入數(shù)據(jù)庫,而不是一直在Session級別緩存。可以考慮設(shè)計(jì)一個(gè)累加器,每保存一個(gè)User實(shí)例,累加器增加1。根據(jù)累加器的值決定是否需要將Session緩存中的數(shù)據(jù)刷入數(shù)據(jù)庫。

          下面是增加100?000個(gè)User實(shí)例的代碼片段:

          1. private void testUser()throws Exception   
          2.   
          3. {   
          4.   
          5.     //打開Session   
          6.   
          7.     Session session = HibernateUtil.currentSession();   
          8.   
          9.     //開始事務(wù)   
          10.   
          11.     Transaction tx = session.beginTransaction();   
          12.   
          13.     //循環(huán)100 000次,插入100 000條記錄   
          14.   
          15.     for (int i = 0 ; i < 1000000 ; i++ )   
          16.     {   
          17.       //創(chuàng)建User實(shí)例   
          18.      User u1 = new User();   
          19.         u1.setName("xxxxx" + i);   
          20.         u1.setAge(i);   
          21.         u1.setNationality("china");   
          22.         //在Session級別緩存User實(shí)例   
          23.         session.save(u1);   
          24.         //每當(dāng)累加器是20的倍數(shù)時(shí),將Session中的數(shù)據(jù)刷入數(shù)據(jù)庫,并清空Session緩存   
          25.   
          26.         if (i % 20 == 0)   
          27.         {   
          28.   
          29.             session.flush();   
          30.   
          31.             session.clear();   
          32.   
          33.             tx.commit();   
          34.   
          35.             tx = session.beginTransaction();   
          36.         }   
          37.     }   
          38.   
          39.     //提交事務(wù)   
          40.     tx.commit();   
          41.       //關(guān)閉事務(wù)   
          42.     HibernateUtil.closeSession();   
          43. }  
          private void testUser()throws Exception
          {
          //打開Session
          Session session = HibernateUtil.currentSession();
          //開始事務(wù)
          Transaction tx = session.beginTransaction();
          //循環(huán)100 000次,插入100 000條記錄
          for (int i = 0 ; i < 1000000 ; i++ )
          {
          //創(chuàng)建User實(shí)例
          User u1 = new User();
          u1.setName("xxxxx" + i);
          u1.setAge(i);
          u1.setNationality("china");
          //在Session級別緩存User實(shí)例
          session.save(u1);
          //每當(dāng)累加器是20的倍數(shù)時(shí),將Session中的數(shù)據(jù)刷入數(shù)據(jù)庫,并清空Session緩存
          if (i % 20 == 0)
          {
          session.flush();
          session.clear();
          tx.commit();
          tx = session.beginTransaction();
          }
          }
          //提交事務(wù)
          tx.commit();
          //關(guān)閉事務(wù)
          HibernateUtil.closeSession();
          }
          

          上面代碼中,當(dāng)i%20 == 0時(shí),手動(dòng)將Session處的緩存數(shù)據(jù)寫入數(shù)據(jù)庫,并手動(dòng)提交事務(wù)。如果不提交事務(wù),數(shù)據(jù)將依然緩存在事務(wù)處——未進(jìn)入數(shù)據(jù)庫,也將引起內(nèi)存溢出的異常。

          這是對Session級別緩存的處理,還應(yīng)該通過如下配置來關(guān)閉SessionFactory的二級      緩存。

          hibernate.cache.use_second_level_cache false

          注意:除了要手動(dòng)清空Session級別的緩存外,最好關(guān)閉SessionFactory級別的二級緩存。否則,即使手動(dòng)清空Session級別的緩存,但因?yàn)樵赟essionFactory級別還有緩存,也可能引發(fā)異常。

          4.2.2 批量更新

          上面介紹的方法同樣適用于批量更新數(shù)據(jù),如果需要返回多行數(shù)據(jù),可以使用scroll()方法,從而可充分利用服務(wù)器端游標(biāo)所帶來的性能優(yōu)勢。下面是進(jìn)行批量更新的代碼片段:

          1. private void testUser()throws Exception   
          2.   
          3. {   
          4.   
          5.     //打開Session   
          6.   
          7.     Session session = HibernateUtil.currentSession();   
          8.   
          9.     //開始事務(wù)   
          10.   
          11.     Transaction tx = session.beginTransaction();   
          12.   
          13.     //查詢出User表中的所有記錄   
          14.   
          15.     ScrollableResults users = session.createQuery("from User")   
          16.   
          17.         .setCacheMode(CacheMode.IGNORE)   
          18.   
          19.         .scroll(ScrollMode.FORWARD_ONLY);   
          20.   
          21.     int count=0;   
          22.   
          23.     //遍歷User表中的全部記錄   
          24.   
          25.     while ( users.next() )   
          26.   
          27.     {   
          28.   
          29.         User u = (User) users.get(0);   
          30.   
          31.         u.setName("新用戶名" + count);   
          32.   
          33.         //當(dāng)count為20的倍數(shù)時(shí),將更新的結(jié)果從Session中flush到數(shù)據(jù)庫   
          34.   
          35.         if ( ++count % 20 == 0 )   
          36.   
          37.         {   
          38.   
          39.             session.flush();   
          40.   
          41.             session.clear();   
          42.   
          43.         }   
          44.   
          45.     }   
          46.   
          47.     tx.commit();   
          48.   
          49.     HibernateUtil.closeSession();   
          50.   
          51. }  
          private void testUser()throws Exception
          {
          //打開Session
          Session session = HibernateUtil.currentSession();
          //開始事務(wù)
          Transaction tx = session.beginTransaction();
          //查詢出User表中的所有記錄
          ScrollableResults users = session.createQuery("from User")
          .setCacheMode(CacheMode.IGNORE)
          .scroll(ScrollMode.FORWARD_ONLY);
          int count=0;
          //遍歷User表中的全部記錄
          while ( users.next() )
          {
          User u = (User) users.get(0);
          u.setName("新用戶名" + count);
          //當(dāng)count為20的倍數(shù)時(shí),將更新的結(jié)果從Session中flush到數(shù)據(jù)庫
          if ( ++count % 20 == 0 )
          {
          session.flush();
          session.clear();
          }
          }
          tx.commit();
          HibernateUtil.closeSession();
          }
          

          通過這種方式,雖然可以執(zhí)行批量更新,但效果非常不好。執(zhí)行效率不高,而且需要先執(zhí)行數(shù)據(jù)查詢,然后再執(zhí)行數(shù)據(jù)更新,并且這種更新將是逐行更新,即每更新一行記錄,都需要執(zhí)行一條update語句,性能非常低下。

          為了避免這種情況,Hibernate提供了一種類似于SQL的批量更新和批量刪除的HQL語法。

          4.2.3 SQL風(fēng)格的批量更新/刪除

          Hibernate提供的HQL語句也支持批量的UPDATE和DELETE語法。

          批量UPDATE和DELETE語句的語法格式如下:

          UPDATE | DELETE FROM? ClassName [WHERE WHERE_CONDITIONS]

          關(guān)于上面的語法格式有以下四點(diǎn)值得注意:

             ● 在FROM子句中,F(xiàn)ROM關(guān)鍵字是可選的。即完全可以不寫FROM關(guān)鍵字。

             ● 在FROM子句中只能有一個(gè)類名,該類名不能有別名。

             ● 不能在批量HQL語句中使用連接,顯式的或隱式的都不行。但可以在WHERE子句中使用子查詢。

             ● 整個(gè)WHERE子句是可選的。

          假設(shè),需要批量更改User類實(shí)例的name屬性,可以采用如下代碼片段完成:

          1. private void testUser()throws Exception   
          2.   
          3. {   
          4.   
          5.     //打開Session   
          6.   
          7.     Session session = HibernateUtil.currentSession();   
          8.   
          9.     //開始事務(wù)   
          10.   
          11.     Transaction tx = session.beginTransaction();   
          12.   
          13.     //定義批量更新的HQL語句   
          14.   
          15.     String hqlUpdate = "update User set name = :newName";   
          16.   
          17.     //執(zhí)行更新   
          18.   
          19.     int updatedEntities = session.createQuery( hqlUpdate )   
          20.   
          21.                            .setString( "newName", "新名字" )   
          22.   
          23.                            .executeUpdate();   
          24.   
          25.     //提交事務(wù)   
          26.   
          27.     tx.commit();   
          28.   
          29.     HibernateUtil.closeSession();   
          30.   
          31. }  
          private void testUser()throws Exception
          {
          //打開Session
          Session session = HibernateUtil.currentSession();
          //開始事務(wù)
          Transaction tx = session.beginTransaction();
          //定義批量更新的HQL語句
          String hqlUpdate = "update User set name = :newName";
          //執(zhí)行更新
          int updatedEntities = session.createQuery( hqlUpdate )
          .setString( "newName", "新名字" )
          .executeUpdate();
          //提交事務(wù)
          tx.commit();
          HibernateUtil.closeSession();
          }
          

          從上面代碼中可以看出,這種語法非常類似于PreparedStatement的executeUpdate語法。實(shí)際上,HQL的這種批量更新就是直接借鑒了SQL語法的UPDATE語句。

          注意:使用這種批量更新語法時(shí),通常只需要執(zhí)行一次SQL的UPDATE語句,就可以完成所有滿足條件記錄的更新。但也可能需要執(zhí)行多條UPDATE語句,這是因?yàn)橛欣^承映射等特殊情況,例如有一個(gè)Person實(shí)例,它有Customer的子類實(shí)例。當(dāng)批量更新Person實(shí)例時(shí),也需要更新Customer實(shí)例。如果采用joined-subclass或union-subclass映射策略,Person和Customer實(shí)例保存在不同的表中,因此可能需要多條UPDATE語句。

          執(zhí)行一個(gè)HQL DELETE,同樣使用 Query.executeUpdate() 方法,下面是一次刪除上面全部記錄的代碼片段:

          1. private void testUser()throws Exception   
          2.   
          3. {   
          4.   
          5.     //打開Session實(shí)例   
          6.   
          7.     Session session = HibernateUtil.currentSession();   
          8.   
          9.     //開始事務(wù)   
          10.   
          11.     Transaction tx = session.beginTransaction();   
          12.   
          13.     //定義批量刪除的HQL語句   
          14.   
          15.     String hqlUpdate = "delete User";   
          16.   
          17.     //執(zhí)行批量刪除   
          18.   
          19.     int updatedEntities = session.createQuery( hqlUpdate )   
          20.   
          21.                            .executeUpdate();   
          22.   
          23.     //提交事務(wù)   
          24.   
          25.     tx.commit();   
          26.   
          27.     //關(guān)閉Session   
          28.   
          29.     HibernateUtil.closeSession();   
          30.   
          31. }  
          private void testUser()throws Exception
          {
          //打開Session實(shí)例
          Session session = HibernateUtil.currentSession();
          //開始事務(wù)
          Transaction tx = session.beginTransaction();
          //定義批量刪除的HQL語句
          String hqlUpdate = "delete User";
          //執(zhí)行批量刪除
          int updatedEntities = session.createQuery( hqlUpdate )
          .executeUpdate();
          //提交事務(wù)
          tx.commit();
          //關(guān)閉Session
          HibernateUtil.closeSession();
          }
          

          由 Query.executeUpdate()方法返回一個(gè)整型值,該值是受此操作影響的記錄數(shù)量。實(shí)際上,Hibernate的底層操作是通過JDBC完成的。因此,如果有批量的UPDATE或DELETE操作被轉(zhuǎn)換成多條UPDATE或DELETE語句,該方法返回的是最后一條SQL語句影響的記錄行數(shù)。

          posted on 2009-10-10 17:26 xiachang88 閱讀(287) 評論(0)  編輯  收藏

          只有注冊用戶登錄后才能發(fā)表評論。


          網(wǎng)站導(dǎo)航:
           
          主站蜘蛛池模板: 巴南区| 北辰区| 隆子县| 大方县| 聂拉木县| 五指山市| 津南区| 玉林市| 阳曲县| 务川| 伊金霍洛旗| 渭南市| 寿阳县| 三门峡市| 东莞市| 基隆市| 榆树市| 湖口县| 安福县| 克东县| 密山市| 孟连| 微山县| 株洲市| 孟村| 惠安县| 额尔古纳市| 嘉荫县| 涞源县| 若尔盖县| 石首市| 北辰区| 雷山县| 汝阳县| 琼中| 浦江县| 攀枝花市| 平顶山市| 班戈县| 鸡西市| 新沂市|