A股上市公司传智教育(股票代码 003032)旗下技术交流社区北京昌平校区

 找回密码
 加入黑马

QQ登录

只需一步,快速开始

© HM代景康 高级黑马   /  2013-10-17 22:43  /  1081 人查看  /  0 人回复  /   0 人收藏 转载请遵从CC协议 禁止商业使用本文

批量更新

  上面介绍的方法同样适用于批量更新数据,如果需要返回多行数据,可以使用scroll()方法,从而可充分利用服务器端游标所带来的性能优势。下面是进行批量更新的代码片段:

  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的倍数时,将更新的结果从Session中flush到数据库

  if ( ++count % 20 == 0 )

  {

  session.flush();

  session.clear();

  }

  }

  tx.commit();

  HibernateUtil.closeSession();

  }

  通过这种方式,虽然可以执行批量更新,但效果非常不好。执行效率不高,而且需要先执行数据查询,然后再执行数据更新,并且这种更新将是逐行更新,即每更新一行记录,都需要执行一条update语句,性能非常低下。

  为了避免这种情况,Hibernate提供了一种类似于SQL的批量更新和批量删除的HQL语法。

  4.2.3 SQL风格的批量更新/删除

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

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

  UPDATE | DELETE FROM? ClassName [WHERE WHERE_CONDITIONS]

  关于上面的语法格式有以下四点值得注意:

  ● 在FROM子句中,FROM关键字是可选的。即完全可以不写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";

  //执行更新

  int updatedEntities = session.createQuery( hqlUpdate )

  .setString( "newName", "新名字" )

  .executeUpdate();

  //提交事务

  tx.commit();

  HibernateUtil.closeSession();

  }

  从上面代码中可以看出,这种语法非常类似于PreparedStatement的executeUpdate语法。实际上,HQL的这种批量更新就是直接借鉴了SQL语法的UPDATE语句。

  注意:使用这种批量更新语法时,通常只需要执行一次SQL的UPDATE语句,就可以完成所有满足条件记录的更新。但也可能需要执行多条UPDATE语句,这是因为有继承映射等特殊情况,例如有一个Person实例,它有Customer的子类实例。当批量更新Person实例时,也需要更新Customer实例。如果采用joined-subclass或union-subclass映射策略,Person和Customer实例保存在不同的表中,因此可能需要多条UPDATE语句。

  执行一个HQL DELETE,同样使用 Query.executeUpdate() 方法,下面是一次删除上面全部记录的代码片段:

  private void testUser()throws Exception

  {

  //打开Session实例

  Session session = HibernateUtil.currentSession();

  //开始事务

  Transaction tx = session.beginTransaction();

  //定义批量删除的HQL语句

  String hqlUpdate = "delete User";

  //执行批量删除

  int updatedEntities = session.createQuery( hqlUpdate )

  .executeUpdate();

  //提交事务

  tx.commit();

  //关闭Session

  HibernateUtil.closeSession();

  }

  由Query.executeUpdate()方法返回一个整型值,该值是受此操作影响的记录数量。实际上,Hibernate的底层操作是通过JDBC完成的。因此,如果有批量的UPDATE或DELETE操作被转换成多条UPDATE或DELETE语句,该方法返回的是最后一条SQL语句影响的记录行数。

评分

参与人数 1黑马币 +3 收起 理由
周志龙 + 3 很给力!

查看全部评分

0 个回复

您需要登录后才可以回帖 登录 | 加入黑马