假设需要批量插入10000条实体数据至数据库。如下是各个操作方法及耗时

环境Spring boot

1、JDBC(JdbcTemplate)
pom.xml
  1. <dependency>
  2. <groupId>org.springframework.boot</groupId>
  3. <artifactId>spring-boot-starter-jdbc</artifactId>
  4. </dependency>
Service
  1. @Autowired
  2. private JdbcTemplate jdbcTemplate;
  3. public void jdbc(List<String> list){
  4. int[] updatedCountArray=jdbcTemplate.batchUpdate("INSERT INTO customer (name) VALUES (?);", new BatchPreparedStatementSetter() {
  5. @Override
  6. public void setValues(PreparedStatement preparedStatement, int i) throws SQLException {
  7. preparedStatement.setString(1,list.get(i));
  8. }
  9. @Override
  10. public int getBatchSize() {
  11. return list.size();
  12. }
  13. });
  14. }

2、网上最常见的JPA----entityManager批量操作方法
Entity
  1. package net.xjdsz.model;
  2. import javax.persistence.*;
  3. /**
  4. * Created by dingshuo on 2017/6/23.
  5. */
  6. @Entity
  7. @Table(name = "customer", schema = "test", catalog = "")
  8. public class CustomerEntity {
  9. private int id;
  10. private String name;
  11. @Id
  12. @GeneratedValue(strategy = GenerationType.AUTO)
  13. @Column(name = "id", nullable = false)
  14. public int getId() {
  15. return id;
  16. }
  17. public void setId(int id) {
  18. this.id = id;
  19. }
  20. @Basic
  21. @Column(name = "name", nullable = true, length = 100)
  22. public String getName() {
  23. return name;
  24. }
  25. public void setName(String name) {
  26. this.name = name;
  27. }
  28. @Override
  29. public boolean equals(Object o) {
  30. if (this == o) return true;
  31. if (o == null || getClass() != o.getClass()) return false;
  32. CustomerEntity that = (CustomerEntity) o;
  33. if (id != that.id) return false;
  34. if (name != null ? !name.equals(that.name) : that.name != null) return false;
  35. return true;
  36. }
  37. @Override
  38. public int hashCode() {
  39. int result = id;
  40. result = 31 * result + (name != null ? name.hashCode() : 0);
  41. return result;
  42. }
  43. }
Service
  1. private EntityManager em;
  2. @PersistenceContext(name = "EntityManagerFactory")
  3. public void SetEntityManager(EntityManager em) {
  4. this.em = em;
  5. }
  6. @Transactional
  7. public void saveBatch(List<CustomerEntity> list) {
  8. for (int i = 0; i < 10000; i++) {
  9. em.persist(list.get(i));
  10. if (i % 1000 == 0) {
  11. em.flush();
  12. em.clear();
  13. }
  14. }
  15. }

3、Jpa---Repository循环写入
Repository
  1. package net.xjdsz.dao;
  2. import net.xjdsz.model.CustomerEntity;
  3. import org.springframework.data.jpa.repository.JpaRepository;
  4. import org.springframework.stereotype.Repository;
  5. /**
  6. * Created by dingshuo on 2017/6/23.
  7. */
  8. @Repository
  9. public interface CustomerRepository extends JpaRepository<CustomerEntity,Integer> {
  10. }

4、Jpa--Repository批量写入
Service
  1. @Transactional
  2. public void saveBatchJpa(List<CustomerEntity> list) {
  3. repository.save(list);
  4. }

实验代码:
用的是Spring boot,所以开了一个RestController去做实验
  1. package net.xjdsz;
  2. import net.xjdsz.dao.CustomerRepository;
  3. import net.xjdsz.dao.TestService;
  4. import net.xjdsz.model.CustomerEntity;
  5. import org.springframework.beans.factory.annotation.Autowired;
  6. import org.springframework.web.bind.annotation.GetMapping;
  7. import org.springframework.web.bind.annotation.RestController;
  8. import java.util.ArrayList;
  9. import java.util.List;
  10. /**
  11. * Created by dingshuo on 2017/6/23.
  12. */
  13. @RestController
  14. public class TestController {
  15. @Autowired
  16. TestService service;//测试用的Service类
  17. @Autowired
  18. CustomerRepository repository; //实体Repository接口
  19. @GetMapping(value = "/test")
  20. public void test(){
  21. List<String> list=new ArrayList<>(); //给jdbctemplate用的集合
  22. List<CustomerEntity> customerEntityList=new ArrayList<>();//给jpa用的集合
  23. for(int i=0;i<10000;i++){
  24. list.add("学生"+i);
  25. CustomerEntity customerEntity=new CustomerEntity();
  26. customerEntity.setName("学生"+i);
  27. customerEntityList.add(customerEntity);
  28. }
  29. //1.jdbc
  30. long startTime=System.currentTimeMillis(); //获取开始时间
  31. service.jdbc(list);
  32. long endTime=System.currentTimeMillis(); //获取结束时间
  33. System.out.println("jdbc程序运行时间: "+(endTime-startTime)+"ms");
  34. //2.jpa-em
  35. long startTime1=System.currentTimeMillis(); //获取开始时间
  36. service.saveBatch(customerEntityList);
  37. long endTime1=System.currentTimeMillis(); //获取结束时间
  38. System.out.println("JPA-EM程序运行时间: "+(endTime1-startTime1)+"ms");
  39. //3.jpa-循环
  40. long startTime2=System.currentTimeMillis(); //获取开始时间
  41. for(int i=0;i<customerEntityList.size();i++){
  42. repository.save(customerEntityList.get(i));
  43. }
  44. long endTime2=System.currentTimeMillis(); //获取结束时间
  45. System.out.println("JPA-循环程序运行时间: "+(endTime2-startTime2)+"ms");
  46. //4.jpa-集合
  47. long startTime3=System.currentTimeMillis(); //获取开始时间
  48. repository.save(customerEntityList);
  49. long endTime3=System.currentTimeMillis(); //获取结束时间
  50. System.out.println("JPA-集合程序运行时间: "+(endTime3-startTime3)+"ms");
  51. }
  52. }

实验结果
  1. jdbc程序运行时间: 878ms
  2. JPA-EM程序运行时间: 2018ms
  3. JPA-循环程序运行时间: 21915ms
  4. JPA-集合程序运行时间: 2373ms

结论就是如果追求极致的性能(批量操作速度),优选JDBC。EM和JPA直接操作集合没有太大的性能区别,这对于新接触Spring JPA(比如我)的人来说,不比纠结有时候没法注入EM,直接使用JPA操作集合即可。
JPA的循环造作相当于对单条insert重复了10000遍,自然最慢,也不推荐了。

如下是JPA操作集合的代码,可以看出起始内部也是用的em,所以可以放心大胆的用了。
  1. @Transactional
  2. public <S extends T> List<S> save(Iterable<S> entities) {
  3. List<S> result = new ArrayList<S>();
  4. if (entities == null) {
  5. return result;
  6. }
  7. for (S entity : entities) {
  8. result.add(save(entity));
  9. }
  10. return result;
  11. }
  12. /*
  13. * (non-Javadoc)
  14. * @see org.springframework.data.repository.CrudRepository#save(java.lang.Object)
  15. */
  16. @Transactional
  17. public <S extends T> S save(S entity) {
  18. if (entityInformation.isNew(entity)) {
  19. em.persist(entity);
  20. return entity;
  21. } else {
  22. return em.merge(entity);
  23. }
  24. }

JPA批量操作及性能比对的更多相关文章

  1. jpa batch批量操作save和persist比较

    1.网上最常见的JPA----entityManager批量操作方法 private EntityManager em; @PersistenceContext(name = "Entity ...

  2. 深入浅出学Spring Data JPA

    第一章:Spring Data JPA入门 Spring Data是什么 Spring Data是一个用于简化数据库访问,并支持云服务的开源框架.其主要目标是使得对数据的访问变得方便快捷,并支持map ...

  3. JPA基础

    目录 目录 1 一.JPA基础 2 1.1 JPA基础 2 1.2JPA开发过程 3 1.3 实体的生命周期及实体管理器常用方法 4 二.环境搭建 5 2.1 添加JPA支持 6 2.2 添加配置文件 ...

  4. Spring Data Jpa配置

    Spring Data JPA提供的接口,也是Spring Data JPA的核心概念: 1:Repository:最顶层的接口,是一个空的接口,目的是为了统一所有Repository的类型,且能让组 ...

  5. JPA && Spring Data && Spring Data JPA

    1.JPA  Java Persistence API,用于对象持久化的一组API,JPA本身是一组规范,让开发者用同一种方式访问不同的ORM框架.其实也就是java实体对象和关系型数据库建立起映射关 ...

  6. JPA学习笔记

    一.JPA基础1.1 JPA基础JPA: java persistence api 支持XML.JDK5.0注解俩种元数据的形式,是SUN公司引入的JPA ORM规范 元数据:对象和表之间的映射关系 ...

  7. Spring Boot 系列教程2-Data JPA

    Spring Data JPA 用来简化创建 JPA 数据访问层和跨存储的持久层功能. 官网文档连接 http://docs.spring.io/spring-data/jpa/docs/curren ...

  8. Spring Data JPA 入门Demo

    什么是JPA呢? 其实JPA可以说是一种规范,是java5.0之后提出来的用于持久化的一套规范:它不是任何一种ORM框架,在我看来,是现有ORM框架在这个规范下去实现持久层. 它的出现是为了简化现有的 ...

  9. Spring Data JPA 初体验

    一,JPA相关的概念 JPA概述 全称是:JavaPersistence API.是SUN公司推出的一套基于ORM的规范. Hibernate框架中提供了JPA的实现. JPA通过JDK 5.0注解或 ...

随机推荐

  1. 洛谷P1541 乌龟棋 [2010NOIP提高组]

    P1541 乌龟棋 题目背景 小明过生日的时候,爸爸送给他一副乌龟棋当作礼物. 题目描述 乌龟棋的棋盘是一行N个格子,每个格子上一个分数(非负整数).棋盘第1格是唯一的起点,第N格是终点,游戏要求玩家 ...

  2. 粉丝裂变活动bug

    1 二维码ios无法扫描 也不知道是现在二维码长按识别的机制改了还是咋样,之前如果二维码ios 太小或者位置不对无法识别就加个透明的二维码,一般是妥妥的,但是这次就是不行,排除fixed,变形等等 解 ...

  3. Spring_使用(JDBC)

    Spring_对JDBC的支持 使用JdbcTemplate更新数据库 导入jar包 创建applicationcontext.xml <?xml version="1.0" ...

  4. oracle-ORA-01650错误

    Unable to extend rollback segment 原因:没有足够的撤销空间用来处理所有活动事务

  5. POJ 1679The Unique MST

    Description Given a connected undirected graph, tell if its minimum spanning tree is unique. Definit ...

  6. JavaScript--开关思想

    就是男默女泪的立flag! <!DOCTYPE html> <html lang="en"> <head> <meta charset=& ...

  7. @atcoder - AGC035F@ Two Histograms

    目录 @description@ @solution@ @accepted code@ @details@ @description@ 给定一个 N*M 的方格,我们通过以下步骤往里面填数: (1)将 ...

  8. Android Tween和Frame 动画

    关于动画的实现,Android提供了Animation,在Android SDK介绍了2种Animation模式: 1. Tween Animation:通过对场景里的对象不断做图像变换(平移.缩放. ...

  9. P2P需集齐四大证照

    今后做P2P需集齐四大证照 比牌照制还严 2016-09-05 11:53:24 分类:热点观察 作者:汪祖刚 8月24日,P2P网贷监管细则在千呼万唤中始出来,整个行业内外的关注热度可谓史无前例.有 ...

  10. Python中多线程与多进程的恩恩怨怨

    概念: 并发:当有多个线程在操作时,如果系统只有一个CPU,则它根本不可能真正同时进行一个以上的线程,它只能把CPU运行时间划分成若干个时间段,再将时间 段分配给各个线程执行,在一个时间段的线程代码运 ...