亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

hibernate批量操作實例詳解

 更新時間:2016年03月24日 10:54:32   作者:新浪網(wǎng)友  
這篇文章主要介紹了hibernate批量操作,結合實例形式分析了Hibernate實現(xiàn)批量插入,更新及刪除等操作的具體實現(xiàn)技巧,需要的朋友可以參考下

本文實例講述了hibernate批量操作的方法。分享給大家供大家參考,具體如下:

Hibernate的批量處理

Hibernate完全以面向對象的方式來操作數(shù)據(jù)庫,當程序里以面向對象的方式操作持久化對象時,將被自動轉換為對數(shù)據(jù)庫的操作。例如調用Session的delete()方法來刪除持久化對象,Hibernate將負責刪除對應的數(shù)據(jù)記錄;當執(zhí)行持久化對象的set方法時,Hibernate將自動轉換為對應的update方法,修改數(shù)據(jù)庫的對應記錄。

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

1  批量插入

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

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();

但隨著這個程序的運行,總會在某個時候運行失敗,并且拋出OutOfMemoryException(內存溢出異常)。這是因為Hibernate的Session持有一個必選的一級緩存,所有的User實例都將在Session級別的緩存區(qū)進行了緩存的緣故。

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

下面是增加100 000個User實例的代碼片段:

private void testUser()throws Exception
{
 //打開Session
 Session session = HibernateUtil.currentSession();
 //開始事務
 Transaction tx = session.beginTransaction();
 //循環(huán)100 000次,插入100 000條記錄
 for (int i = 0 ; i < 1000000 ; i++ )
 {
  //創(chuàng)建User實例
  User u1 = new User();
  u1.setName("xxxxx" + i);
  u1.setAge(i);
  u1.setNationality("china");
  //在Session級別緩存User實例
  session.save(u1);
  //每當累加器是20的倍數(shù)時,將Session中的數(shù)據(jù)刷入數(shù)據(jù)庫,并清空Session緩存
  if (i % 20 == 0)
  {
   session.flush();
   session.clear();
   tx.commit();
   tx = session.beginTransaction();
  }
 }
 //提交事務
 tx.commit();
 //關閉事務
 HibernateUtil.closeSession();
}

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

這是對Session級別緩存的處理,還應該通過如下配置來關閉SessionFactory的二級緩存。

hibernate.cache.use_second_level_cache false

注意:除了要手動清空Session級別的緩存外,最好關閉SessionFactory級別的二級緩存。否則,即使手動清空Session級別的緩存,但因為在SessionFactory級別還有緩存,也可能引發(fā)異常。

2  批量更新

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

private void testUser()throws Exception
{
 //打開Session
 Session session = HibernateUtil.currentSession();
 //開始事務
 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);
  //當count為20的倍數(shù)時,將更新的結果從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語法。

3  SQL風格的批量更新/刪除

Hibernate提供的HQL語句也支持批量的UPDATE和DELETE語法。
批量UPDATE和DELETE語句的語法格式如下:

UPDATE | DELETE FROM? ClassName [WHERE WHERE_CONDITIONS]

關于上面的語法格式有以下四點值得注意:

● 在FROM子句中,F(xiàn)ROM關鍵字是可選的。即完全可以不寫FROM關鍵字。
● 在FROM子句中只能有一個類名,該類名不能有別名。
● 不能在批量HQL語句中使用連接,顯式的或隱式的都不行。但可以在WHERE子句中使用子查詢。
● 整個WHERE子句是可選的。

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

private void testUser()throws Exception
{
 //打開Session
 Session session = HibernateUtil.currentSession();
 //開始事務
 Transaction tx = session.beginTransaction();
 //定義批量更新的HQL語句
 String hqlUpdate = "update User set name = :newName";
 //執(zhí)行更新
 int updatedEntities = session.createQuery( hqlUpdate )
       .setString( "newName", "新名字" )
       .executeUpdate();
 //提交事務
 tx.commit();
 HibernateUtil.closeSession();
}

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

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

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

private void testUser()throws Exception
{
 //打開Session實例
 Session session = HibernateUtil.currentSession();
 //開始事務
 Transaction tx = session.beginTransaction();
 //定義批量刪除的HQL語句
 String hqlUpdate = "delete User";
 //執(zhí)行批量刪除
 int updatedEntities = session.createQuery( hqlUpdate )
       .executeUpdate();
 //提交事務
 tx.commit();
 //關閉Session
 HibernateUtil.closeSession();
}

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

希望本文所述對大家基于Hibernate框架的Java程序設計有所幫助。

相關文章

  • Java括號匹配舉例詳解

    Java括號匹配舉例詳解

    看到大家對括號匹配問題很感興趣,下面這篇文章主要給大家介紹了關于Java括號匹配的相關資料,括號匹配是一種常見的編程問題,涉及到在給定的字符串中判斷括號是否匹配,需要的朋友可以參考下
    2023-10-10
  • Java通過工廠、Map容器創(chuàng)建對象的方法

    Java通過工廠、Map容器創(chuàng)建對象的方法

    這篇文章主要介紹了Java通過工廠、Map容器創(chuàng)建對象的方法,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-03-03
  • idea同時編輯多行問題-win&mac都支持

    idea同時編輯多行問題-win&mac都支持

    這篇文章主要介紹了idea同時編輯多行問題-win&mac都支持,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2023-09-09
  • Java實現(xiàn)簡單局域網(wǎng)聊天室

    Java實現(xiàn)簡單局域網(wǎng)聊天室

    這篇文章主要為大家詳細介紹了Java實現(xiàn)簡單局域網(wǎng)聊天室,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-06-06
  • Springboot項目如何兼容老的Spring項目問題

    Springboot項目如何兼容老的Spring項目問題

    這篇文章主要介紹了Springboot項目如何兼容老的Spring項目問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2023-08-08
  • springboot整合vue實現(xiàn)上傳下載文件

    springboot整合vue實現(xiàn)上傳下載文件

    這篇文章主要為大家詳細介紹了springboot整合vue實現(xiàn)上傳下載文件,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-11-11
  • Java 類型信息詳解和反射機制介紹

    Java 類型信息詳解和反射機制介紹

    這篇文章主要介紹了Java 類型信息詳解和反射機制介紹,需要的朋友可以參考下
    2020-11-11
  • Map集合之HashMap的使用及說明

    Map集合之HashMap的使用及說明

    這篇文章主要介紹了Map集合之HashMap的使用及說明,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-10-10
  • 詳解log4j-over-slf4j與slf4j-log4j12共存stack overflow異常分析

    詳解log4j-over-slf4j與slf4j-log4j12共存stack overflow異常分析

    這篇文章主要介紹了詳解log4j-over-slf4j與slf4j-log4j12共存stack overflow異常分析,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-07-07
  • JAVA對象JSON數(shù)據(jù)互相轉換的四種常見情況

    JAVA對象JSON數(shù)據(jù)互相轉換的四種常見情況

    這篇文章主要介紹了JAVA對象JSON數(shù)據(jù)互相轉換的四種常見情況,需要的朋友可以參考下
    2014-04-04

最新評論