线程分批处理数据及MyBatis的批量插入

文章目录
一、背景
二、代码实现:
三、分页查询下游批次处理场景
四、MyBatis的批量插入
1、活动表简单表结构:
2、业务层组装数据:

一、背景
数据量较多时,我们常常遇到需要分批处理的情况,比如上千上万数据需要需要操作数据库时(入库或者更新),我们想到分批处理,或者解析文件数据量较多,我们可能多线程分批处理,每个线程执行一定的批次数据等…
不管何种场景,当我们想到分批处理时,那如何分批,怎么计算批次数呢?


tip:此处以小数据量为例:假设待处理数据总条数75,每10条数据一处理

我常用的方法总结两种供参考:

方法一:循环遍历待处理的处理,当到达待处理的批次或批次倍数,或最后不足一次剩余的,满足条件即执行一次分批处理操作。eg:10,20,30…7
方法二:根据总数计算总批次数(75/10+1=8次),每次从待处理数据集中截取要处理的数据,最后一个批次特殊处理。eg:0-9,10-19…70-74

二、代码实现:

public static void main(String[] args) {
        List sourList = new ArrayList();
        //1、将list中放进去75个数据--模拟待处理数据集
        for (int i = 0; i < 75; i++) {
            sourList.add(i);
        }

        //假设每N个(10个)数据一处理(实际应用中可能每N个开一个线程,或者每N个调一次数据库操作)
        int batchSize = 10;
        //tempList存放批次数据
        List<Object> tempList = new ArrayList<Object>();

        //方法一:循环遍历待处理的处理(可能多线程,可能分批次入数据库,总之需要按每次处理的大小,计算需要处理的批次数)
        for (int i = 0; i < sourList.size(); i++) {
            tempList.add(sourList.get(i));
            //判断批次,够N个数开始处理一波
            if ((i + 1) % batchSize == 0 || (i + 1) == sourList.size()) {
                log.info("处理数据集:{}", tempList.toString());
                //todo 你的线程处理,或者数据库操作等
                tempList.clear();
            }
        }
        log.info("************* 方法一结束 *************");

        //方法二:根据总数计算总批次数,每次从待处理数据集中截取要处理的数据
        int count = sourList.size();
        int totalPages = (count % batchSize > 0) ? (count / batchSize + 1) : (count / batchSize);
        log.info("待处理数据总条数:{},总批次数:{}", count, totalPages);

        int startIndex = 0;
        int endIndex = 0;
        if (totalPages > 0) {
            //循环批次数,计算待处理数据下标
            for (int pageNum = 1; pageNum <= totalPages; pageNum++) {
                //每批次计算要处理的数据起始位置。终止位置
                List subList = new ArrayList();
                startIndex = ((pageNum - 1) * batchSize);
                if (pageNum == totalPages) {
                    //考虑最后一个批次特殊处理
                    subList = sourList.subList(startIndex, count);
                } else {
                    endIndex = startIndex + batchSize;
                    subList = sourList.subList(startIndex, endIndex);
                }
                //todo 你的线程处理,或者数据库操作等
                log.info("处理第{}批次:{}", pageNum, subList.toString());
            }
        }
        log.info("************* 方法二结束 *************");
    }

========================打印结果:

17:42:03.357  处理数据集:[0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
17:42:03.368  处理数据集:[10, 11, 12, 13, 14, 15, 16, 17, 18, 19]
17:42:03.368  处理数据集:[20, 21, 22, 23, 24, 25, 26, 27, 28, 29]
17:42:03.368  处理数据集:[30, 31, 32, 33, 34, 35, 36, 37, 38, 39]
17:42:03.368  处理数据集:[40, 41, 42, 43, 44, 45, 46, 47, 48, 49]
17:42:03.368  处理数据集:[50, 51, 52, 53, 54, 55, 56, 57, 58, 59]
17:42:03.368  处理数据集:[60, 61, 62, 63, 64, 65, 66, 67, 68, 69]
17:42:03.368  处理数据集:[70, 71, 72, 73, 74]
17:42:03.368  ************* 方法一结束 *************
17:42:03.368  待处理数据总条数:75,总批次数:8
17:42:03.369  处理第1批次:[0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
17:42:03.369  处理第2批次:[10, 11, 12, 13, 14, 15, 16, 17, 18, 19]
17:42:03.369  处理第3批次:[20, 21, 22, 23, 24, 25, 26, 27, 28, 29]
17:42:03.369  处理第4批次:[30, 31, 32, 33, 34, 35, 36, 37, 38, 39]
17:42:03.369  处理第5批次:[40, 41, 42, 43, 44, 45, 46, 47, 48, 49]
17:42:03.369  处理第6批次:[50, 51, 52, 53, 54, 55, 56, 57, 58, 59]
17:42:03.369  处理第7批次:[60, 61, 62, 63, 64, 65, 66, 67, 68, 69]
17:42:03.369  处理第8批次:[70, 71, 72, 73, 74]
17:42:03.369  ************* 方法二结束 *************

这样每一波分批待处理的数据集合,以及位置你都能get到了,也就可以进行你的业务逻辑操作了,入库或者开线程了。

三、分页查询下游批次处理场景
如果是分页查询下游或者其他接口循环处理场景:

//每次导出或者查询下游接口等分页页大小(10页一查)
public final static int  PERSIZE = 10;
//查询第一次数据(返回的分页信息page中包含总页数、总条数)
Page<OrderInfoDTO> firstBOList = orderRepository.queryPrescriptionByOrderId(param);
//处理第一页查询回来的数据
if (CollectionUtil.isNotEmpty(firstBOList)) {
    log.info("处理数据{}条 ", firstBOList.size());
}
//页数大于1继续查询
int totalPages=firstBOList.getTotalPage();
if (totalPages > 1) {
    for (int pageNum = 2; pageNum <= totalPages; pageNum++) {
        query.setPageSize(PERSIZE);
        query.setPageNo(pageNum);
        Page<OrderInfoDTO> list = orderRepository.queryPrescriptionByOrderId(param);
        //todo 处理结果集
        log.info("处理第{}批次数据{}条 ", pageNum, list.size());
    }
}
// 每次导出或者查询下游接口等分页页大小(10页一查)
public final static int PERSIZE = 10;

// 查询第一次数据(返回的分页信息page中包含总页数、总条数)
Page<OrderInfoDTO> firstBOList = orderRepository.queryPrescriptionByOrderId(param);

// 处理第一页查询回来的数据
if (CollectionUtil.isNotEmpty(firstBOList)) {
    log.info("处理数据{}条 ", firstBOList.size());
    // 逐条插入相应表
    for (OrderInfoDTO order : firstBOList) {
        // 插入相应表的操作
        // insertIntoTable(order);
    }
}

// 页数大于1继续查询
int totalPages = firstBOList.getTotalPage();
if (totalPages > 1) {
    for (int pageNum = 2; pageNum <= totalPages; pageNum++) {
        query.setPageSize(PERSIZE);
        query.setPageNo(pageNum);
        Page<OrderInfoDTO> list = orderRepository.queryPrescriptionByOrderId(param);
        // 处理结果集
        if (CollectionUtil.isNotEmpty(list)) {
            log.info("处理第{}批次数据{}条 ", pageNum, list.size());
            // 逐条插入相应表
            for (OrderInfoDTO order : list) {
                // 插入相应表的操作
                // insertIntoTable(order);
            }
        }
    }
}


根据以上代码,查询 SQL 和插入 SQL 的编写示例如下:

查询 SQL 示例:

// 假设 param 是查询条件对象
String sql = "SELECT * FROM order_info WHERE order_id = :orderId";
List<OrderInfoDTO> result = jdbcTemplate.query(sql, new BeanPropertySqlParameterSource(param), new BeanPropertyRowMapper<>(OrderInfoDTO.class));
插入 SQL 示例:

// 假设 order 是要插入的数据对象
String sql = "INSERT INTO target_table (column1, column2, column3) VALUES (:value1, :value2, :value3)";
Map<String, Object> paramMap = new HashMap<>();
paramMap.put("value1", order.getValue1());
paramMap.put("value2", order.getValue2());
paramMap.put("value3", order.getValue3());
jdbcTemplate.update(sql, paramMap);

四、MyBatis的批量插入
当我们有很多数据需要通过数据库操作,例如很多数据的更新或者插入, 如果用for循环的方式,里面一条条数据去调数据库操作,我们知道那样开销是很大的,会频繁的创建链接,带来性能问题,为了减少连接,我们通常会批量操作。

批量插入代码样例,已mybatis的批量插入活动表为例:

1、活动表简单表结构:

CREATE TABLE `activity` (
  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT '自增id',
  `activity_id` bigint(20) DEFAULT NULL COMMENT '活动id',
  `user_id` bigint(20) DEFAULT NULL COMMENT '用户id',
  `create_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间',
  PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT DEFAULT CHARSET=utf8 COMMENT='活动表';

2、业务层组装数据:
ActivityServiceImpl.java:


private void saveRecords() {
        List<Activity> recordList =  new ArrayList<>();
            Activity activity = new Activity();
            activity.setUserId(userId);
            activity.setActivityId(activityId);
            recordList.add(activity);
            //此处根据业务需求具体组装待处理集合list
           activityDao.insertBatchActivity(recordList);
    }

activityDao.xml中写法:

activityDao.xml:

 <!-- 批量插入活动表-->
    <insert id="insertBatchActivity">
        INSERT INTO activity
        (
        activity_id, user_id, create_time
        )
        VALUES
        <foreach collection="list" item="item" separator=",">
            (
            #{item.activityId},
            #{item.userId},
            now()
            )
        </foreach>
    </insert>

特别注意:mysql默认接受sql的大小是1048576(1M),即第三种方式若数据量超过1M会报如下异常:(可通过调整MySQL安装目录下的my.ini文件中[mysqld]段的"max_allowed_packet = 1M")

posted @ 2024-01-29 16:57  我的心儿  阅读(665)  评论(0编辑  收藏  举报