SEO教程

SEO教程

Products

当前位置:首页 > SEO教程 >

MyBatis批量插入,5分钟到3秒,优化三步走?

96SEO 2026-04-22 22:13 1


说实话,上周那个数据迁移的需求差点让我崩溃。要把老系统里那十万条用户数据一股脑全导进新库,本来以为是个小case,结果第一版代码跑起来进度条走得比我蜗牛还慢——整整5分钟!kan着日志里那一行行慢吞吞的输出,我就在想,这dou2024年了难道我们的数据库还在用算盘算数吗?

MyBatis批量插入,5分钟到3秒,优化三步走?

领导路过工位,瞥了一眼屏幕,眉头皱得Neng夹死苍蝇:“这么点数据要跑半天?Neng不Neng快点?”

那一刻,我深知必须得动真格的了。经过一下午的折腾,从SQL语句拼接到JDBC参数调优,再到多线程并发,硬是把时间压缩到了3秒。这中间的坑和经验,我觉得有必要好好复盘一下毕竟谁也不想半夜被运维 第一阶段:告别“愚蠢”的循环,拥抱拼接SQL

咱们先来kankanZui原始的写法。坦白讲,这种代码我见过太多次了甚至有些几年经验的“老鸟”为了省事也这么写。

// 方式1:循环单条插入
for  {
    userMapper.insert;
}

这有什么问题?问题大了去了!你想想,10万条数据,意味着你的应用要和数据库进行10万次网络握手,数据库要解析10万次SQL语句,还要开启和提交10万次事务。假设每次交互耗时3毫秒,光网络往返和解析就要300秒,也就是5分钟。这还是在数据库没压力的情况下要是稍微有点并发,这时间还得翻倍。

于是我想到了MyBatis的``标签,把一大堆数据拼成一条巨无霸SQL。



    INSERT INTO user  VALUES
    
    

代码逻辑上也Zuo个分批处理,别一次性把10万条dou塞进去,那样SQL字符串长得吓人。

int batchSize = 1000;
for ; i += batchSize) {
    int end = Math.min);
    List batch = userList.subList;
    userMapper.batchInsert;
}

这一招下去,效果立竿见影,时间直接从5分钟降到了30秒。提升了10倍,感觉是不是Yi经Ke以了?

别高兴太早。这里有个巨大的隐患:`max_allowed_packet`。这是MySQL的一个配置参数,限制了单个SQL包的Zui大长度。Ru果你一次拼个几千条数据进去,SQL长度一旦超过这个限制,直接报错。而且,这种超长的SQL,数据库解析起来也费劲,内存占用高得吓人。

第二阶段:开启JDBC的“外挂”,真正的批处理

30秒虽然比5分钟强多了但面对十万级数据,还是显得不够kan。这时候,就得祭出JDBC驱动层面的黑科技了。

hen多人不知道,MySQL的JDBC驱动里藏着一个参数:rewriteBatchedStatements。这个参数默认是false的,一旦你把它设为true,奇迹就会发生。

修改你的数据库连接URL,加上这行关键配置:

jdbc:mysql://localhost:3306/test?rewriteBatchedStatements=true

光改URL还不够,MyBatis的执行器模式也得换。默认的`SIMPLE`模式是一条条发的,我们需要切换到`BATCH`模式。这时候,别再用那个``拼接大SQL了回归到单条插入的写法,但是用批处理模式去执行。

@Autowired
private SqlSessionFactory sqlSessionFactory;
public void batchInsertWithExecutor {
    // 开启BATCH模式,第二个参数false表示不自动提交
    try ) {
        UserMapper mapper = sqlSession.getMapper;
        int batchSize = 1000;
        for ; i++) {
            mapper.insert);
            // 每满1000条,手动刷新一次防止内存溢出
            if  % batchSize == 0) {
                sqlSession.flushStatements;
                sqlSession.clearCache;
            }
        }
        // 刷新剩余的
        sqlSession.flushStatements;
        sqlSession.commit;
    }
}

这背后的原理其实挺有意思。在`ExecutorType.BATCH`模式下MyBatis会先把SQL语句缓存起来而不是立马发给数据库。配合`rewriteBatchedStatements=true`,MySQL驱动会在底层偷偷把这些`INSERT`语句重写成`INSERT INTO table VALUES , , `的形式。

这相当于我们既享受了代码层面循环插入的简单,又拿到了拼接SQL的高性Neng。而且,驱动帮我们控制了包的大小,不会触发`max_allowed_packet`限制。

这一套组合拳打下来耗时直接从30秒降到了8秒。这速度,Yi经Neng应付大部分场景了。

第三阶段:多线程并发,榨干CPU性Neng

但是咱们Zuo技术的,不就是要追求极致吗?8秒虽然快,但kan着日志里那单线程的跑法,我就觉得亏——现在的服务器动不动就是8核16核,你只用一个线程干活,剩下的7个dou在那儿摸鱼,这Neng忍?

既然单线程Neng跑8秒,那Ru果开4个线程并发跑,理论上是不是Neng到2秒?

说干就干。我们把这10万条数据切分成几份,扔进线程池里并行处理。不过这里有个坑,得注意一下:数据库连接池的大小限制了你的并发上限。别线程开了一堆,结果dou在等数据库连接,那就适得其反了。

public void parallelBatchInsert {
    // 线程数别瞎开,得根据数据库连接池来定,比如8个
    int threadCount = 8; 
    int batchSize = userList.size / threadCount;
    ExecutorService executor = Executors.newFixedThreadPool;
    List futures = new ArrayList<>;
    for  {
        int start = i * batchSize;
        int end =  ? userList.size :  * batchSize;
        List subList = userList.subList;
        futures.add -> {
            // 每个线程跑一个批处理任务
            batchInsertWithExecutor;
        }));
    }
    // 等待所有任务dou干完
    for  {
        try {
            future.get;
        } catch  {
            throw new RuntimeException;
        }
    }
    executor.shutdown;
}

这一步加上去,效果简直炸裂。从8秒直接干到了3秒!kan着那瞬间完成的数据导入,心里那叫一个舒坦。

终极方案:整合与细节

为了方便以后直接复用,我把这些逻辑封装成了一个Service。这里用到了`CountDownLatch`来确保所有线程dou跑完了再结束方法,比`future.get`稍微优雅那么一点点。

@Service
public class BatchInsertService {
    @Autowired
    private SqlSessionFactory sqlSessionFactory;
    /**
     * 高性Neng批量插入
     * 10万条数据实测约3秒
     */
    public void highPerformanceBatchInsert {
        if ) {
            return;
        }
        // 线程数控制,别超过CPU核心数太多
        int threadCount = Math.min.availableProcessors);
        int batchSize =  Math.ceil userList.size / threadCount);
        ExecutorService executor = Executors.newFixedThreadPool;
        CountDownLatch latch = new CountDownLatch;
        for  {
            int start = i * batchSize;
            int end = Math.min * batchSize, userList.size);
            if ) {
                latch.countDown;
                continue;
            }
            // 注意这里要new一个新的ArrayList,防止并发修改问题
            List subList = new ArrayList<>);
            executor.submit -> {
                try {
                    doBatchInsert;
                } finally {
                    latch.countDown;
                }
            });
        }
        try {
            latch.await;
        } catch  {
            Thread.currentThread.interrupt;
        }
        executor.shutdown;
    }
    private void doBatchInsert {
        try ) {
            UserMapper mapper = sqlSession.getMapper;
            for ; i++) {
                mapper.insert);
                // 每隔1000条刷新一次平衡内存与速度
                if  % 1000 == 0) {
                    sqlSession.flushStatements;
                    sqlSession.clearCache;
                }
            }
            sqlSession.flushStatements;
            sqlSession.commit;
        }
    }
}
还有几个不得不提的“坑”

虽然速度上去了但实际生产环境用的时候,还有几个细节得盯紧了不然出了问题排查起来Neng让你头秃。

1. 内存溢出风险

我刚才的例子是假设10万条数据Neng一次性加载到内存。但Ru果是千万级数据,你直接`List all = selectAll`,服务器的内存条估计要冒烟。这时候得用流式查询,或者分页读取。比如每次从老库查1万条,插完再查下一批,别贪多。

int pageSize = 10000;
int total = countTotal;
for  {
    List page = selectByPage;
    highPerformanceBatchInsert;
}

2. 自增主键的获取

Ru果你用了`useGeneratedKeys="true"`想要回填主键ID,那得小心了。开启`rewriteBatchedStatements=true`时老版本的MySQL驱动在返回自增ID时会出问题,可Neng会乱套或者报错。解决办法hen简单:把MySQL驱动升级到8.0以上的版本。


    ...

3. 事务管理

多线程环境下事务管理变得hen复杂。上面的代码里每个线程dou有自己的SqlSession和事务。Ru果中间某个线程挂了怎么回滚?这得kan你的业务需求。Ru果是全量覆盖,那失败重跑就行;Ru果是必须保证原子性,可Neng就得考虑分布式事务或者放弃多线程,退而求然后用单线程批处理。毕竟鱼和熊掌不可兼得嘛。

一下

回顾一下这场从5分钟到3秒的“极速之旅”,核心其实就三板斧:

优化阶段 关键动作 耗时变化
初始状态 for循环单条插入 5分钟
第一步 foreach拼接SQL + 分批提交 30秒
第二步 rewriteBatchedStatements + ExecutorType.BATCH 8秒
第三步 多线程并行处理 3秒

其实优化的本质无非就是:减少网络交互次数降低数据库解析开销充分利用计算资源。只要抓住了这三点,不管你是用MyBatis还是JPA,思路dou是通用的。

当然实际项目中别为了快而快,还得考虑代码的可维护性、出错率以及服务器的承受Neng力。不过当老板下次再问你“Neng不Neng再快点”的时候,你至少Neng掏出这几招镇场子了。

Ru果你在批量操作中还遇到过什么奇葩的坑,或者有geng骚的优化技巧,欢迎在评论区留言,咱们一起交流一下毕竟独乐乐不如众乐乐嘛!


标签: 件事

SEO优化服务概述

作为专业的SEO优化服务提供商,我们致力于通过科学、系统的搜索引擎优化策略,帮助企业在百度、Google等搜索引擎中获得更高的排名和流量。我们的服务涵盖网站结构优化、内容优化、技术SEO和链接建设等多个维度。

百度官方合作伙伴 白帽SEO技术 数据驱动优化 效果长期稳定

SEO优化核心服务

网站技术SEO

  • 网站结构优化 - 提升网站爬虫可访问性
  • 页面速度优化 - 缩短加载时间,提高用户体验
  • 移动端适配 - 确保移动设备友好性
  • HTTPS安全协议 - 提升网站安全性与信任度
  • 结构化数据标记 - 增强搜索结果显示效果

内容优化服务

  • 关键词研究与布局 - 精准定位目标关键词
  • 高质量内容创作 - 原创、专业、有价值的内容
  • Meta标签优化 - 提升点击率和相关性
  • 内容更新策略 - 保持网站内容新鲜度
  • 多媒体内容优化 - 图片、视频SEO优化

外链建设策略

  • 高质量外链获取 - 权威网站链接建设
  • 品牌提及监控 - 追踪品牌在线曝光
  • 行业目录提交 - 提升网站基础权威
  • 社交媒体整合 - 增强内容传播力
  • 链接质量分析 - 避免低质量链接风险

SEO服务方案对比

服务项目 基础套餐 标准套餐 高级定制
关键词优化数量 10-20个核心词 30-50个核心词+长尾词 80-150个全方位覆盖
内容优化 基础页面优化 全站内容优化+每月5篇原创 个性化内容策略+每月15篇原创
技术SEO 基本技术检查 全面技术优化+移动适配 深度技术重构+性能优化
外链建设 每月5-10条 每月20-30条高质量外链 每月50+条多渠道外链
数据报告 月度基础报告 双周详细报告+分析 每周深度报告+策略调整
效果保障 3-6个月见效 2-4个月见效 1-3个月快速见效

SEO优化实施流程

我们的SEO优化服务遵循科学严谨的流程,确保每一步都基于数据分析和行业最佳实践:

1

网站诊断分析

全面检测网站技术问题、内容质量、竞争对手情况,制定个性化优化方案。

2

关键词策略制定

基于用户搜索意图和商业目标,制定全面的关键词矩阵和布局策略。

3

技术优化实施

解决网站技术问题,优化网站结构,提升页面速度和移动端体验。

4

内容优化建设

创作高质量原创内容,优化现有页面,建立内容更新机制。

5

外链建设推广

获取高质量外部链接,建立品牌在线影响力,提升网站权威度。

6

数据监控调整

持续监控排名、流量和转化数据,根据效果调整优化策略。

SEO优化常见问题

SEO优化一般需要多长时间才能看到效果?
SEO是一个渐进的过程,通常需要3-6个月才能看到明显效果。具体时间取决于网站现状、竞争程度和优化强度。我们的标准套餐一般在2-4个月内开始显现效果,高级定制方案可能在1-3个月内就能看到初步成果。
你们使用白帽SEO技术还是黑帽技术?
我们始终坚持使用白帽SEO技术,遵循搜索引擎的官方指南。我们的优化策略注重长期效果和可持续性,绝不使用任何可能导致网站被惩罚的违规手段。作为百度官方合作伙伴,我们承诺提供安全、合规的SEO服务。
SEO优化后效果能持续多久?
通过我们的白帽SEO策略获得的排名和流量具有长期稳定性。一旦网站达到理想排名,只需适当的维护和更新,效果可以持续数年。我们提供优化后维护服务,确保您的网站长期保持竞争优势。
你们提供SEO优化效果保障吗?
我们提供基于数据的SEO效果承诺。根据服务套餐不同,我们承诺在约定时间内将核心关键词优化到指定排名位置,或实现约定的自然流量增长目标。所有承诺都会在服务合同中明确约定,并提供详细的KPI衡量标准。

SEO优化效果数据

基于我们服务的客户数据统计,平均优化效果如下:

+85%
自然搜索流量提升
+120%
关键词排名数量
+60%
网站转化率提升
3-6月
平均见效周期

行业案例 - 制造业

  • 优化前:日均自然流量120,核心词无排名
  • 优化6个月后:日均自然流量950,15个核心词首页排名
  • 效果提升:流量增长692%,询盘量增加320%

行业案例 - 电商

  • 优化前:月均自然订单50单,转化率1.2%
  • 优化4个月后:月均自然订单210单,转化率2.8%
  • 效果提升:订单增长320%,转化率提升133%

行业案例 - 教育

  • 优化前:月均咨询量35个,主要依赖付费广告
  • 优化5个月后:月均咨询量180个,自然流量占比65%
  • 效果提升:咨询量增长414%,营销成本降低57%

为什么选择我们的SEO服务

专业团队

  • 10年以上SEO经验专家带队
  • 百度、Google认证工程师
  • 内容创作、技术开发、数据分析多领域团队
  • 持续培训保持技术领先

数据驱动

  • 自主研发SEO分析工具
  • 实时排名监控系统
  • 竞争对手深度分析
  • 效果可视化报告

透明合作

  • 清晰的服务内容和价格
  • 定期进展汇报和沟通
  • 效果数据实时可查
  • 灵活的合同条款

我们的SEO服务理念

我们坚信,真正的SEO优化不仅仅是追求排名,而是通过提供优质内容、优化用户体验、建立网站权威,最终实现可持续的业务增长。我们的目标是与客户建立长期合作关系,共同成长。

提交需求或反馈

Demand feedback