Mybatis之批量更新数据(批量update)

article/2025/10/21 9:23:32

前言

当我们使用mybatis的时候,可能经常会碰到一批数据的批量更新问题,因为如果一条数据一更新,那每一条数据就需要涉及到一次数据库的操作,包括网络IO以及磁盘IO,可想而知,这个效率是非常低下的。而平时我们很少直接使用原生jdbc直接操作数据库,而是会使用比较成熟的ORM框架,那么今天我们就来总结一下,如何使用mybatis做批量更新。

方案一(个人推荐)

在mybatis的xml文件中,使用foreach动态标签拼接SQL语句,每一条数据的更新语句对应一条update语句,多条语句最终使用";"号进行拼接。

<update id="updateBatchById"><foreach collection="list" item="item" separator=";">update`t_student`set`name` = #{item.name},`age` = #{item.age}whereid = #{item.id}</foreach>
</update>

测试执行的时候,通过控制台打印,我们可以很方便地看到上述代码最终生成的SQL语句就是按照我们的本意是由";"号拼接的语句串。

在这里插入图片描述
这条SQL长串会一次性发送给数据库执行,只需要进行一次网络IO,提高了效率。当然,默认情况下,数据库是不支持执行这样由";“号拼接的长串的,执行的时候会报错,提示说执行的SQL有语法错误
在这里插入图片描述
我们需要通过在数据库连接URL中指定allowMultiQueries参数值为true告诉数据库以支持”;"号分隔的多条语句的执行。

spring.datasource.url=jdbc:mysql://localhost:3306/test?allowMultiQueries=true

方案二(个人不推荐)

同样是使用foreach动态标签拼接SQL语句,但是这种方案是最终拼接成一条SQL,而不是多条SQL的长串。

update `t_student`
<trim prefix="set" suffixOverrides=","><trim prefix=" `name` = case " suffix=" end, "><foreach collection="list" item="item"><if test="item.name != null">when `id` = #{item.id} then #{item.name}</if></foreach></trim><trim prefix=" `age` = case " suffix=" end, "><foreach collection="list" item="item"><if test="item.age != null">when `id` = #{item.id} then #{item.age}</if></foreach></trim>
</trim>
where`id` in
<foreach collection="list" item="item" open="(" close=")" separator=",">#{item.id}
</foreach>

最终控制台打印出的SQL如下

UPDATE `t_student` 
SET `name` =
CASEWHEN `id` = 1 THEN'张三' WHEN `id` = 2 THEN'李四' WHEN `id` = 3 THEN'王五' WHEN `id` = 4 THEN'赵六' END,`age` =
CASEWHEN `id` = 1 THEN40 WHEN `id` = 2 THEN34 WHEN `id` = 3 THEN55 WHEN `id` = 4 THEN76 END 
WHERE`id` IN ( 1, 2, 3, 4 )

可以发现,如果批量数据量太大,要更新的字段太多,那这个SQL就会非常难看且复杂,充斥大量的case when判断,而且这种case when感觉也会增大数据库压力,因为这种case when都需要数据库自己去做判断,所以个人感觉不太好,所以不推荐。


http://chatgpt.dhexx.cn/article/PlSnqU96.shtml

相关文章

List和Set之间的转换:达到集合元素去重复

在项目中看到一段代码&#xff1a; return new ArrayList<>(new HashSet<LabelEnum>(list)); 当时我的表情是这样的&#xff1a; 这把list作为构造参数创建新的集合来返回是要做什么&#xff0c;这里的list也是ArrayList集合&#xff0c;这样绕了一圈返回…

【JAVA】List转Set并按照List的顺序排序,HashSet、LinkedHashSet、TreeSet元素保存顺序List转换对比

话不多说 直接代码测试效果: // 简化代码 直接 数组转list 就不用 写很多add了 哈哈哈 String[] array {"f","a","r","q","b","e","o","z","v","p","g"};//…

java中Stream的使用以及List转set和map方法

Stream流的作用&#xff1a; 非常精简方便的去遍历集合实现过滤&#xff0c;排序等。 图解&#xff1a; 例&#xff1a; ArrayList<User> users new ArrayList<>();User user1 new User("赵六",33);users.add(new User("张三",12));users…

JAVA创建对象全过程详解

java世界里面对象无处不在&#xff0c;拿在创建对象的时候都经过哪些步骤&#xff1f; 总结下来大概分为这几步。 对象创建的过程(new 对象的时候) 判断类有没有被加载如果没有(就开始加载类(就是类的加载过程))初始化 &#xff1a;就是给一些变量进行初始化。设置对象头(比较…

Java对象的创建和使用

目录 1、类和对象 2、类的定义方式 3、对象的创建语法 4、什么是实例变量&#xff1f; 5、引用传递的本质 6、引用既可以是局部变量&#xff0c;也可以是成员变量。 7、在一个类的方法中&#xff0c;也可以new本类。 8、NullPointerException&#xff08;空指针异常&…

Java创建对象的方式

Java创建对象的五种方式&#xff1a; &#xff08;1&#xff09;使用new关键字 &#xff08;2&#xff09;使用Object类的clone方法 &#xff08;3&#xff09;使用Class类的newInstance方法 &#xff08;4&#xff09;使用Constructor类中的newInstance方法 &#xff08;5&am…

Java创建对象的几种方式

java是一种面向对象语言,所以我们在写代码过程中会创建很多对象,那java创建的对象到底有多少种呢?其中每种的差别又有哪些呢?请允许我慢慢道来 1.使用new关键字 这是最常见也是使用最多的一种。 Test test = new Test();// 无参构造函数 如果我们想要在创建对象的时候,…

java 之创建对象

文章目录 前言创建对象new关键字的作用构造方法什么是初始化构造方法的作用构造方法的两种形式参考引用 前言 这是我学习过程中做的总结&#xff0c;如有不对见谅。 创建对象 我们用Demo类来创建一个对象。 Demo demonew Demo();这一条语句&#xff0c;其实包括了四个动作&a…

Java中创建对象的5种方法

将会列举5种方法去创建 Java 对象&#xff0c;以及他们如何与构造函数交互&#xff0c;并且会有介绍如何去使用这些方法的示例。 作为一个 Java 开发人员&#xff0c;我们每天都会创建大量的 Java 对象&#xff0c;但是我们通常会使用依赖管理系统去创建这些对象&#xff0c;例…

Java创建对象的四种方式

1. new 2. clone 3. 通过反射newInstance 4. 反序列化 5. String s “abc”&#xff08;这个是比较特殊的&#xff09; 以String类为例 String string null; Class class1 String.class;// 该方法最为安全可靠&#xff0c;程序性能更高。 Class class2 string.getClass(…

c# Topshelf创建linux与Windows服务

目录 Topshelf安装Topshelf包代码如下&#xff0c;简单粗暴卸载服务安装服务 Topshelf 讨厌创建.net 服务时的窗体怎么办&#xff1f;讨厌调试.net服务怎么办?调试.net服务还要自己建控制台怎么办? Topshelf 它来了&#xff01;&#xff01;&#xff01; Topshelf 是一个开源…

htop与top命令

安装htop yum -y install htop htop 类似于 top 命令&#xff0c;但可以让你在垂直和水平方向上滚动&#xff0c;所以你可以看到系统上运行的所有进程&#xff0c;以及他们完整的命令行。可以不用输入进程的 PID 就可以对此进程进行相关的操作 (killing, renicing)。 与 Lin…

使用Quartz.net实现多线程任务定时执行,动态配置Job,结合Topshelf构建Windows服务

几个月前有这么个需求&#xff1a;需要执行一些Job&#xff0c;这些Job会各自按照不同的时间频次执行&#xff0c;且它们做的事情也不同&#xff0c;有的是监控站点&#xff0c;有的是监控服务器存储情况&#xff0c;有的是监控报表PROCEDURE的执行状况… OK&#xff0c;当看到…

top命令参数详解

简介 top命令是Linux下常用的性能分析工具&#xff0c;能够实时显示系统中各个进程的资源占用状况&#xff0c;类似于Windows的任务管理器。 top显示系统当前的进程和其他状况,是一个动态显示过程,即可以通过用户按键来不断刷新当前状态.如果在前台执行该命令,它将独占前台,直…

top命令参数详解(linux top命令的用法详细详解)

通过top命令可以有效的发现系统的缺陷出在哪里。是内存不够、CPU处理能力不够、IO读写过高。 top命令输出长这样&#xff1a; top命令参数详解&#xff08;linux top命令的用法详细详解&#xff09; 以下解析一下各个字段的意思&#xff1a; VIRT&#xff1a;virtual memory …

Topas命令详解

执行topas命令后如图所示&#xff1a; #topas 操作系统的最全面动态&#xff0c;而又查看方便的性能视图就是topas命令了&#xff0c;下面以topas输出为例&#xff0c;对AIX系统的性能监控做简要描述&#xff0c;供运维工程师和系统管理员们参考。 另&#xff1a;1.操作系统报…

Topshelf 打包部署Windows服务

1 创建项目(例&#xff1a;控制台程序) Nuget 引入Topshelf类库 using System; using System.Threading; using System.Threading.Tasks; using Topshelf;namespace LoginTypeInherit {public class Program{private static readonly log4net.ILog log log4net.LogManager.G…

Linux top命令参数详解

Linux top命令参数详解 生产环境系统运行慢&#xff0c;出现无法响应通常原因主要还在于分析CPU、内存、磁盘使用率情况&#xff0c;并结合命令查找出具体进程&#xff0c;并在进程中进一步分析主要因子情况&#xff0c;渗透到对于其中包含线程占用情况的分析。一般而言对于ja…

C#之TopShelf启动Windows服务

写了一两天&#xff0c;才发现组长给的原始代码原本就有Topshelf&#xff0c;还是写出来提示我topshelf不明确哪个版本的使用&#xff0c;莫名尴尬。 1、项目的主要运行代码 HostFactory.Run(x >{x.RunAsLocalSystem();x.SetDescription("topshelf测试");x.SetDi…

Linux下top命令用法详解

一、命令介绍 Linux top命令用于实时显示 process &#xff08;进程&#xff09;的动态。它用于监控正在运行系统负荷的信息&#xff0c;包括系统负载、CPU利用分布情况、内存使用、每个进程的资源占用情况等。 使用权限&#xff1a;所有使用者 二、命令详解 在命令行下输入…