Tips

原文作者:Vlad Mihalcea

原文地址:The best way to do batch processing with JPA and Hibernate

在本文中,你将了解什么是批处理,为什么要使用它,以及如何在JPA和Hibernate中正确使用它。

批处理

在编写企业应用程序时,通常将工作分配在服务于典型OLTP(在线事务处理)传输的前端系统,和一个或多个批处理用于ETL(Extract,Transform,Load)操作。

批处理器,顾名思义,将要处理的数据分成几个块,因此具有以下优点:

  • 每个块可以通过单独的工作线程进行处理,因此增加了吞吐量并减少了整个处理时间。
  • 每个块都可以使用自己的数据库事务,所以如果有一个失败,就不用丢掉我们所做的所有工作,只是针对当前事务的变化。

JPA 批处理

当使用JPA时,假设要插入50个Post实体,那么应该这样做:

int entityCount = 50;
int batchSize = 25;
 
EntityManager entityManager = null;
EntityTransaction transaction = null;
 
try {
    entityManager = entityManagerFactory()
        .createEntityManager();
 
    transaction = entityManager.getTransaction();
    transaction.begin();
 
    for ( int i = 0; i < entityCount; ++i ) {
        if ( i > 0 && i % batchSize == 0 ) {
            entityManager.flush();
            entityManager.clear();
 
            transaction.commit();
            transaction.begin();
        }
 
        Post post = new Post(
            String.format( "Post %d", i + 1 )
        );
        entityManager.persist( post );
    }
 
    transaction.commit();
} catch (RuntimeException e) {
    if ( transaction != null &&
         transaction.isActive()) {
        transaction.rollback();
    }
    throw e;
} finally {
    if (entityManager != null) {
        entityManager.close();
    }
}

每个开始操作都会开启事务,因为每个实体状态转换必须在数据库事务的范围内执行。

for循环一次会持久化一个Post 实体。 但是,由于实体状态转换仅在执行flush方法更新数据库时执行,因此我们可以将多个SQL 插入语句分组到到要给单一PreparedStatement执行中,该执行需要多个参数。

每次迭代计数器(变量i)已达到batchSize临界值的倍数,我们可以刷新EntityManager并提交数据库事务。 通过在每次批处理执行后提交数据库事务,我们获得以下优点:

  1. 避免了长期运行的事务,这对MVCC关系数据库系统是不利的。
  2. 我们确保如果执行失败,不会丢失以前成功执行的批处理作业完成的工作。

EntityManager在每次批量执行后被清除,这样就不会继续累积可能导致管理实体的几个问题:

  1. 如果要持久化的实体数量庞大,那么存在内存不足的风险。
  2. 在持久化上下文中累积的实体越多,flush越慢。 所以,最好的做法是确保持久性上下文尽可能的短小。

如果抛出异常,我们必须确保回滚当前正在运行的数据库事务。 否则可能会导致许多问题,因为数据库可能仍然认为事务处于打开状态,锁可能会被持有,直到事务超时或由DBA结束。

最后,我们需要关闭EntityManager,以便可以清除上下文并释放Session级的资源。

虽然这是使用JPA进行批处理的正确方法,但还没有完成。 如前所述,也可以从JDBC批量更新中受益。 为此,我们需要提供以下Hibernate配置属性:

<property
    name="hibernate.jdbc.batch_size"
    value="25"
/>
 
<property
    name="hibernate.order_inserts"  
    value="true"
/>
 
<property
    name="hibernate.order_updates"  
    value="true"
/>

这些属性允许我们将多个SQL语句批处理为单个PreparedStatement执行,这需要单个数据库往返。 选择值25以匹配EntityManager批处理作业的临界值。

执行以前的测试用例时,Hibernate生成2个SQL INSERT语句:

INSERT INTO post (title, id) values (?, ?)"],
Params:[
    (Post 1, 1), (Post 2, 2), (Post 3, 3),
    (Post 4, 4), (Post 5, 5), (Post 6, 6),
    (Post 7, 7), (Post 8, 8), (Post 9, 9),
    (Post 10, 10), (Post 11, 11), (Post 12, 12),
    (Post 13, 13), (Post 14, 14), (Post 15, 15),
    (Post 16, 16), (Post 17, 17), (Post 18, 18),
    (Post 19, 19), (Post 20, 20), (Post 21, 21),
    (Post 22, 22), (Post 23, 23), (Post 24, 24),
    (Post 25, 25)
]
 
INSERT INTO post (title, id) values (?, ?)"],
Params:[
    (Post 26, 26), (Post 27, 27), (Post 28, 28),
    (Post 29, 29), (Post 30, 30), (Post 31, 31),
    (Post 32, 32), (Post 33, 33), (Post 34, 34),
    (Post 35, 35), (Post 36, 36), (Post 37, 37),
    (Post 38, 38), (Post 39, 39), (Post 40, 40),
    (Post 41, 41), (Post 42, 42), (Post 43, 43),
    (Post 44, 44), (Post 45, 45), (Post 46, 46),
    (Post 47, 47), (Post 48, 48), (Post 49, 49),
    (Post 50, 50)
]

结论

了解如何正确设计批处理作业在开发企业应用程序时非常重要。 幸运的是,使用JPA和Hibernate,这个任务很容易实现。 最重要的方面是考虑到如何让关系数据库工作效果最好,这就是驱动数据访问设计决策的因素。

使用JPA和Hibernate进行批量处理的最佳方式的更多相关文章

  1. JPA入门例子(采用JPA的hibernate实现版本) 转

    JPA入门例子(采用JPA的hibernate实现版本) jpahibernate数据库jdbcjava框架(1).JPA介绍: JPA全称为Java Persistence API ,Java持久化 ...

  2. JPA和Hibernate的相关使用技巧

    介绍 尽管有SQL标准,但每个关系数据库终将是唯一的,因此你需要调整数据访问层,以便充分利用在使用中的关系数据库. 在本文中,我们将介绍在使用带有JPA和Hibernate的MySQL时,为了提高性能 ...

  3. JPA入门例子(采用JPA的hibernate实现版本) --- 会伴随 配置文件:persistence.xml

    JPA入门例子(采用JPA的hibernate实现版本) 分类: j2se2011-03-30 16:09 45838人阅读 评论(9) 收藏 举报 jpahibernate数据库jdbcjava框架 ...

  4. Hibernate的批量插入(&&JDBC)

    来自: http://blog.csdn.net/an_2016/article/details/51759890 一.批量插入(两种方式) 1,通过hibernate缓存 如果这样写代码进行批量插入 ...

  5. JPA和hibernate的关系

    实际上,JPA的标准的定制是hibernate作者参与定制的,所以JPA是hibernate的一个总成,可以这么理解

  6. JPA与Hibernate的关系

    1.JPA JPA全称: Java Persistence API  JPA通过JDK 5.0注解或XML描述对象-关系表的映射关系,并将运行期的实体对象持久化到数据库中.  JPA的出现?  JPA ...

  7. JPA和Hibernate的区别

    JPA Java Persistence API,是Java EE 5的标准ORM接口,也是ejb3规范的一部分. Hibernate,当今很流行的ORM框架,是JPA的一个实现,但是其功能是JPA的 ...

  8. java(样品集成框架spring、spring mvc、spring data jpa、hibernate)

    这是你自己的参考springside集成框架的开源项目.主要的整合spring.spring mvc.spring data jpa.hibernate几个框架,对于这些框架中仍然感觉更舒适sprin ...

  9. 谈论Hibernate级联删除——JPA根据Hibernate实现许多级联删除CascadeType.DELETE_ORPHAN

    声明: 1.这篇文章是原创.非复制或转载过来. 2.在本文中,参数都亲自做过实验证明. 3.这篇文章谈到了Hibernate配置基于注释的方法.hbm语法不可用. 不清JPA.Hibernate.EJ ...

随机推荐

  1. systemtap原理及使用

    SystemTap的架构 SystemTap用于检查运行的内核的两种方法是 Kprobes和 返回探针.但是理解任何内核的最关键要素是内核的映射,它提供符号信息(比如函数.变量以及它们的地址).有了内 ...

  2. Linux 上搭建 git 的服务器

    搭建服务器 假设服务器的名字是 git.example.com. 首先,添加一个叫做git的用户adduser git. 然后如果不存在的话, 为这个用户新建一个主目录mkdir /home/git, ...

  3. OC中的单例

    概念 单例模式的意图是类的对象称为系统中唯一的实例,提供一个访问点,供客户类共享资源 什么情况下使用单例 )类只能由一个实例,而且必须从一个为人熟知的访问点对其进行访问,比如工厂方法 )这个唯一的实例 ...

  4. ⑤JS返回格式化的当前时间和上周时间

    首先对时间进行格式化 返回上周时间和当前时间

  5. hdu 2114 Calculate S(n) 数论(简单题)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2114 自己对数论一窍不通啊现在,做了一道水题,贴出来吧...主要是让自己记住这个公式: 前n项和的立方 ...

  6. html5,js插件实现手机端实现头像剪切上传

    思路:先打开相册,选取图片,在剪切图片,转化为base64格式,然后上传到七牛存储,返回url,再传给后端,整个流程就是这样.用的是angular框架,图像插件用到imagecropper.js,废话 ...

  7. New Adventure----GUI Design Studio

    新建项目工程    File->New Project 新建设计文件    Project->New Design 单个设计文件的页面,F9运当前设计页面   页面控件中有绿色包围的控件为 ...

  8. Java UDP Socket

    本文转载地址:          Java学习路上的收获:http://blog.csdn.net/qinpeng100423/article/details/8980423 一. UDP协议定义 U ...

  9. OpenCV探索之路(十四):绘制点、直线、几何图形

    绘制点和圆 void cvCircle( CvArr* img, CvPoint center, int radius, CvScalar color, int thickness=1, int li ...

  10. 第 9 章 MySQL数据库Schema设计的性能优化

    前言: 很多人都认为性能是在通过编写代码(程序代码或者是数据库代码)的过程中优化出来的,其实这是一个非常大的误区.真正影响性能最大的部分是在设计中就已经产生了的,后期的优化很多时候所能够带来的改善都只 ...