• easyexcel结合jdbc手动控制事务加批处理实现百万数据导入


    1. @Resource
    2. private EmpService empService;
    3. @GetMapping("/importData")
    4. public void importData() {
    5. String fileName = "C:\\Users\\asus\\Desktop\\员工信息.xlsx";
    6. //记录开始读取Excel时间,也是导入程序开始时间
    7. long startReadTime = System.currentTimeMillis();
    8. System.out.println("------开始读取Excel的Sheet时间(包括导入数据过程):" + startReadTime + "ms------");
    9. //读取所有Sheet的数据.每次读完一个Sheet就会调用这个方法
    10. EasyExcel.read(fileName, new EasyExceGeneralDatalListener(empService)).doReadAll();
    11. long endReadTime = System.currentTimeMillis();
    12. System.out.println("------结束读取Excel的Sheet时间(包括导入数据过程):" + endReadTime + "ms------");
    13. System.out.println("------读取Excel的Sheet时间(包括导入数据)共计耗时:" + (endReadTime-startReadTime) + "ms------");
    14. }

    Excel导入事件监听

    1. // 事件监听
    2. public class EasyExceGeneralDatalListener extends AnalysisEventListener> {
    3. /**
    4. * 处理业务逻辑的Service,也可以是Mapper
    5. */
    6. private EmpService empService;
    7. /**
    8. * 用于存储读取的数据
    9. */
    10. private List> dataList = new ArrayList>();
    11. public EasyExceGeneralDatalListener() {
    12. }
    13. public EasyExceGeneralDatalListener(EmpService empService) {
    14. this.empService = empService;
    15. }
    16. @Override
    17. public void invoke(Map data, AnalysisContext context) {
    18. //数据add进入集合
    19. dataList.add(data);
    20. //size是否为100000条:这里其实就是分批.当数据等于10w的时候执行一次插入
    21. if (dataList.size() >= ExcelConstants.GENERAL_ONCE_SAVE_TO_DB_ROWS) {
    22. //存入数据库:数据小于1w条使用Mybatis的批量插入即可;
    23. saveData();
    24. //清理集合便于GC回收
    25. dataList.clear();
    26. }
    27. }
    28. /**
    29. * 保存数据到DB
    30. *
    31. * @param
    32. * @MethodName: saveData
    33. * @return: void
    34. */
    35. private void saveData() {
    36. empService.importData(dataList);
    37. dataList.clear();
    38. }
    39. /**
    40. * Excel中所有数据解析完毕会调用此方法
    41. *
    42. * @param: context
    43. * @MethodName: doAfterAllAnalysed
    44. * @return: void
    45. */
    46. @Override
    47. public void doAfterAllAnalysed(AnalysisContext context) {
    48. saveData();
    49. dataList.clear();
    50. }
    51. }
    核心业务代码
    1. public interface EmpService {
    2. void export() throws IOException;
    3. void importData(List> dataList);
    4. }
    5. /*
    6. * 测试用Excel导入超过10w条数据,经过测试发现,使用Mybatis的批量插入速度非常慢,所以这里可以使用 数据分批+JDBC分批插入+事务来继续插入速度会非常快
    7. */
    8. @Override
    9. public void importData(List> dataList) {
    10. //结果集中数据为0时,结束方法.进行下一次调用
    11. if (dataList.size() == 0) {
    12. return;
    13. }
    14. //JDBC分批插入+事务操作完成对20w数据的插入
    15. Connection conn = null;
    16. PreparedStatement ps = null;
    17. try {
    18. long startTime = System.currentTimeMillis();
    19. System.out.println(dataList.size() + "条,开始导入到数据库时间:" + startTime + "ms");
    20. conn = JDBCDruidUtils.getConnection();
    21. //控制事务:默认不提交
    22. conn.setAutoCommit(false);
    23. String sql = "insert into emp (`empno`, `ename`, `job`, `mgr`, `hiredate`, `sal`, `comm`, `deptno`) values";
    24. sql += "(?,?,?,?,?,?,?,?)";
    25. ps = conn.prepareStatement(sql);
    26. //循环结果集:这里循环不支持lambda表达式
    27. for (int i = 0; i < dataList.size(); i++) {
    28. Map item = dataList.get(i);
    29. ps.setString(1, item.get(0));
    30. ps.setString(2, item.get(1));
    31. ps.setString(3, item.get(2));
    32. ps.setString(4, item.get(3));
    33. ps.setString(5, item.get(4));
    34. ps.setString(6, item.get(5));
    35. ps.setString(7, item.get(6));
    36. ps.setString(8, item.get(7));
    37. //将一组参数添加到此 PreparedStatement 对象的批处理命令中。
    38. ps.addBatch();
    39. }
    40. //执行批处理
    41. ps.executeBatch();
    42. //手动提交事务
    43. conn.commit();
    44. long endTime = System.currentTimeMillis();
    45. System.out.println(dataList.size() + "条,结束导入到数据库时间:" + endTime + "ms");
    46. System.out.println(dataList.size() + "条,导入用时:" + (endTime - startTime) + "ms");
    47. } catch (Exception e) {
    48. e.printStackTrace();
    49. } finally {
    50. //关连接
    51. JDBCDruidUtils.close(conn, ps);
    52. }
    53. }
    54. }
    jdbc工具类
    1. //JDBC工具类
    2. public class JDBCDruidUtils {
    3. private static DataSource dataSource;
    4. /*
    5. 创建数据Properties集合对象加载加载配置文件
    6. */
    7. static {
    8. Properties pro = new Properties();
    9. //加载数据库连接池对象
    10. try {
    11. //获取数据库连接池对象
    12. pro.load(JDBCDruidUtils.class.getClassLoader().getResourceAsStream("druid.properties"));
    13. dataSource = DruidDataSourceFactory.createDataSource(pro);
    14. } catch (Exception e) {
    15. e.printStackTrace();
    16. }
    17. }
    18. /*
    19. 获取连接
    20. */
    21. public static Connection getConnection() throws SQLException {
    22. return dataSource.getConnection();
    23. }
    24. /**
    25. * 关闭conn,和 statement独对象资源
    26. *
    27. * @param connection
    28. * @param statement
    29. * @MethodName: close
    30. * @return: void
    31. */
    32. public static void close(Connection connection, Statement statement) {
    33. if (connection != null) {
    34. try {
    35. connection.close();
    36. } catch (SQLException e) {
    37. e.printStackTrace();
    38. }
    39. }
    40. if (statement != null) {
    41. try {
    42. statement.close();
    43. } catch (SQLException e) {
    44. e.printStackTrace();
    45. }
    46. }
    47. }
    48. /**
    49. * 关闭 conn , statement 和resultset三个对象资源
    50. *
    51. * @param connection
    52. * @param statement
    53. * @param resultSet
    54. * @MethodName: close
    55. * @return: void
    56. */
    57. public static void close(Connection connection, Statement statement, ResultSet resultSet) {
    58. close(connection, statement);
    59. if (resultSet != null) {
    60. try {
    61. resultSet.close();
    62. } catch (SQLException e) {
    63. e.printStackTrace();
    64. }
    65. }
    66. }
    67. /*
    68. 获取连接池对象
    69. */
    70. public static DataSource getDataSource() {
    71. return dataSource;
    72. }
    73. }

    druid.properties配置文件

    这里我将文件创建在类路径下,需要注意的是连接mysql数据库时需要指定rewriteBatchedStatements=true批处理才会生效,否则还是逐条插入效率较低,allowMultiQueries=true表示可以使sql语句中有多个insert或者update语句(语句之间携带分号),这里可以忽略。

    1. # druid.properties配置
    2. driverClassName=com.mysql.jdbc.Driver
    3. url=jdbc:mysql://localhost:3306/llp?autoReconnect=true&useUnicode=true&useSSL=false&serverTimezone=GMT%2B8&allowMultiQueries=true&rewriteBatchedStatements=true
    4. username=root
    5. password=root
    6. initialSize=10
    7. maxActive=50
    8. maxWait=60000

    测试结果

    1. ------开始读取Excel的Sheet时间(包括导入数据过程):1674181403555ms------
    2. 200000条,开始导入到数据库时间:1674181409740ms
    3. 2023-01-20 10:23:29.943 INFO 18580 --- [nio-8888-exec-1] com.alibaba.druid.pool.DruidDataSource : {dataSource-1} inited
    4. 200000条,结束导入到数据库时间:1674181413252ms
    5. 200000条,导入用时:3512ms
    6. 200000条,开始导入到数据库时间:1674181418422ms
    7. 200000条,结束导入到数据库时间:1674181420999ms
    8. 200000条,导入用时:2577ms
    9. .....
    10. 200000条,开始导入到数据库时间:1674181607405ms
    11. 200000条,结束导入到数据库时间:1674181610154ms
    12. 200000条,导入用时:2749ms
    13. ------结束读取Excel的Sheet时间(包括导入数据过程):1674181610155ms------
    14. ------读取Excel的Sheet时间(包括导入数据)共计耗时:206600ms------

  • 相关阅读:
    代码随想录算法训练营第23期day59|503.下一个更大元素II、42. 接雨水
    【云原生之Docker实战】使用Docker部署jpress开源网站
    【大数据之Kafka】十二、Kafka之offset位移及漏消费和重复消费
    什么是EVM?以太坊EVM合约交互
    前端开发学习之【AJAX】
    循环链表的设计与基本操作的实现
    数据存储(二)WebStorage 之 IndexedDB
    【模型训练】YOLOv7车辆三类别检测
    新建 ASP.NET MVC 三层项目 | 新建 ASP.NET MVC 项目
    离散化模板
  • 原文地址:https://blog.csdn.net/weixin_59244784/article/details/136781629