mybatis 批量 update,mybatis foreach批量更新
00-1010 1.更新多条数据,每条数据都不一样。1.逐个更新(java实现)2。逐个更新(mybatis实现)3。sql的批量更新(映射/实现)4。批量更新(单字段,参考列表),其实是SQL批量更新的简化版。5.SQL的批量更新(通过插入实现)2。多次更新。
00-1010背景描述:一般情况下,如果需要一次性更新多条数据,有两种方式:(1)循环遍历业务代码,逐个更新。(2)一次性更新所有数据(更准确的说,一条sql语句更新所有数据,逐个更新的操作放在数据库端,业务代码端显示一次性更新所有数据)。这两种方法各有利弊。下面将简要分析两种方法的优缺点,主要介绍第二种方法在mybatis中的实现。
00-1010这个方法显然是最简单的,也是最不容易出错的。即使发生错误,也只是影响当前错误的数据,对于每一条数据都是可以控制的。如果更新失败或成功,可以从逻辑代码中获得。代码可能如下所示:
updateBatch(ListMyData datas){ for(MyData data : datas){ try { MyData Dao . update(data);//更新一段数据,比如mybatis {.//如果更新失败,可以做一些其他的操作,比如,打印错误日志等。} } }//my batis中更新操作的实现更新我的数据集.在哪里./更新这种方法最大的问题是效率。它一个个更新,每次连接数据库,然后更新,然后释放连接资源(虽然连接池可以大大提高频繁连接数据的效率,无法抵御数据量大)。这种损失在数据量较大时会体现出效率问题。这也是为了满足业务需求,通常采用上述的二次批量更新的实现方式(当然这种方式也有数据大小的限制,后面会提到)。
00-1010通过一个循环依次执行多个sql更新。
条件:要实现批量更新,必须设置mysql支持批量操作,jdbc链接必须附加属性allowMultiQueries=true。
例如:
JDBC : my SQL ://localhost :3306/dbname?character encoding=ut F8 zerodatetimebehavior=converttullallowmultiqueries=true update id= updateBatch parameter type= Java . util . list foreach collection= list item= item index= index open= close= separator=;update course set name=$ { item . name }/set where id=$ { item . id }/foreach/update一条记录更新一次,性能差,容易造成拥塞。
00-1010 (1)、实战练习(传入的ListMapString,Object)一定要注意,必须加上where条件,里面的id是要更新的数据的id;如果没有添加where条件,则全部更新,但需要更新且有数据的更新是传递的数据,没有数据的更新为null。这时,更新错误发生了。
update id= updatechartparambyaccountandhartid parameter type= list update follow me _ parameters trim prefix= set suffix overrides=, trim prefix= signal _ source=case suffix= end,
<foreach collection="list" item="item" index="index"> <if test="item.signalSource!=null"> when account=#{item.account} and chart_id=#{item.chartId} then #{item.signalSource} </if> </foreach> </trim> <trim prefix="rate =case" suffix="end,"> <foreach collection="list" item="item" index="index"> <if test="item.rate!=null"> when account=#{item.account} and chart_id=#{item.chartId} then #{item.rate} </if> </foreach> </trim> </trim> where id in <foreach collection="list" item="item" index="index" separator="," open="(" close=")"> #{item.id} </foreach> </update>另外文章的样板
<update id="updateBatch" parameterType="list"> update course <trim prefix="set" suffixOverrides=","> <trim prefix="peopleId =case" suffix="end,"> <foreach collection="list" item="i" index="index"> <if test="i.peopleId!=null"> when id=#{i.id} then #{i.peopleId} </if> </foreach> </trim> <trim prefix=" roadgridid =case" suffix="end,"> <foreach collection="list" item="i" index="index"> <if test="i.roadgridid!=null"> when id=#{i.id} then #{i.roadgridid} </if> </foreach> </trim> <trim prefix="type =case" suffix="end," > <foreach collection="list" item="i" index="index"> <if test="i.type!=null"> when id=#{i.id} then #{i.type} </if> </foreach> </trim> <trim prefix="unitsid =case" suffix="end," > <foreach collection="list" item="i" index="index"> <if test="i.unitsid!=null"> when id=#{i.id} then #{i.unitsid} </if> </foreach> </trim> </trim> where <foreach collection="list" separator="or" item="i" index="index" > id=#{i.id} </foreach></update>
(2)、下面逐步讲解
一条sql语句来批量更新所有数据,下面直接看一下在mybatis中通常是怎么写的(去掉mybatis语法就是原生的sql语句了,所有就没单独说sql是怎么写的)。
<update id="updateBatch" parameterType="java.util.List"> update mydata_table set status= <foreach collection="list" item="item" index="index" separator=" " open="case ID" close="end"> when #{item.id} then #{item.status} </foreach> where id in <foreach collection="list" index="index" item="item" separator="," open="(" close=")"> #{item.id,jdbcType=BIGINT} </foreach> </update>
其中when...then...是sql中的"switch" 语法。这里借助mybatis的语法来拼凑成了批量更新的sql,上面的意思就是批量更新id在updateBatch参数所传递List中的数据的status字段。还可以使用实现同样的功能,代码如下:
<update id="updateBatch" parameterType="java.util.List"> update mydata_table <trim prefix="set" suffixOverrides=","> <trim prefix="status =case" suffix="end,"> <foreach collection="list" item="item" index="index"> when id=#{item.id} then #{item.status} </foreach> </trim> </trim> where id in <foreach collection="list" index="index" item="item" separator="," open="(" close=")"> #{item.id,jdbcType=BIGINT} </foreach> </update><trim>
属性说明
1.prefix,suffix 表示在trim标签包裹的部分的前面或者后面添加内容2.如果同时有prefixOverrides,suffixOverrides 表示会用prefix,suffix覆盖Overrides中的内容。3.如果只有prefixOverrides,suffixOverrides 表示删除开头的或结尾的xxxOverides指定的内容。
上述代码转化成sql如下:
update mydata_table set status = case when id = #{item.id} then #{item.status}//此处应该是<foreach>展开值 ... end where id in (...);
当然这是最简单的批量更新实现,有时候可能需要更新多个字段,那就需要将
<trim prefix="status =case" suffix="end,"> <foreach collection="list" item="item" index="index"> when id=#{item.id} then #{item.status} </foreach></trim>
复制拷贝多次,更改prefix和when...then...的内容即可.而如果当需要为某个字段设置默认值的时候可以使用else
<trim prefix="status =case" suffix="end,"> <foreach collection="list" item="item" index="index"> when id=#{item.id} then #{item.status} </foreach> else default_value</trim>
还有更常见的情况就是需要对要更新的数据进行判断,只有符合条件的数据才能进行更新,这种情况可以这么做:
<trim prefix="status =case" suffix="end,"> <foreach collection="list" item="item" index="index"> <if test="item.status !=null and item.status != -1"> when id=#{item.id} then #{item.status} </if> </foreach></trim>
这样的话只有要更新的list中status != null && status != -1的数据才能进行status更新.其他的将使用默认值更新,而不会保持原数据不变.如果要保持原数据不变呢?即满足条件的更新,不满足条件的保持原数据不变,简单的来做就是再加一个,因为mybatis中没有if...else...语法,但可以通过多个实现同样的效果,如下:
<trim prefix="status =case" suffix="end,"> <foreach collection="list" item="item" index="index"> <if test="item.status !=null and item.status != -1"> when id=#{item.id} then #{item.status} </if> <if test="item.status == null or item.status == -1"> when id=#{item.id} then mydata_table.status //这里就是原数据 </if> </foreach></trim>
整体批量更新的写法如下:
<update id="updateBatch" parameterType="java.util.List"> update mydata_table <trim prefix="set" suffixOverrides=","> <trim prefix="status =case" suffix="end,"> <foreach collection="list" item="item" index="index"> <if test="item.status !=null and item.status != -1"> when id=#{item.id} then #{item.status} </if> <if test="item.status == null or item.status == -1"> when id=#{item.id} then mydata_table.status//原数据 </if> </foreach> </trim> </trim> where id in <foreach collection="list" index="index" item="item" separator="," open="(" close=")"> #{item.id,jdbcType=BIGINT} </foreach></update>
4.批量更新(单个字段,传参list),实际是sql批量更新的简化版本而已
(1)、单个字段方法1
<update id="updateByBatch" parameterType="java.util.List"> update t_goods set NODE_ID= <foreach collection="list" item="item" index="index" separator=" " open="case" close="end"> when GOODS_ID=#{item.goodsId} then #{item.nodeId} </foreach> where GOODS_ID in <foreach collection="list" index="index" item="item" separator="," open="(" close=")"> #{item.goodsId,jdbcType=BIGINT} </foreach> </update>
(2)、单个字段方法2
<update id="updateByBatch" parameterType="java.util.List"> UPDATE t_goods SET NODE_ID = CASE <foreach collection="list" item="item" index="index"> WHEN GOODS_ID = #{item.goodsId} THEN #{item.nodeId} </foreach> END WHERE GOODS_ID IN <foreach collection="list" index="index" item="item" open="(" separator="," close=")"> #{item.goodsId} </foreach> </update>
以上单字段更新实际执行:
UPDATE t_goods SET NODE_ID = CASE WHEN GOODS_ID = ? THEN ? END WHERE GOODS_ID IN ( ? )
5.sql批量更新(通过insert实现)
传入的是List<Map<String,Object>>
直接运行插入,如果有插入的数据转为更新该条数据
<insert id="updateChartParamByAccountAndChartid"> insert into followme_parameters (account,chart_id,signal_source,rate) values <foreach collection="list" separator="," index="index" item="item"> (#{item.account},#{item.chartId},#{item.signalSource},#{item.rate}) </foreach> ON duplicate KEY UPDATE signal_source=values(signal_source),rate=values(rate) </insert>
二.更新多条数据,更新的内容一样.
1.传map/传String
NODE_ID从map中取出来,goodsIdList是字符串拼接好的(如下面的"1,2,5")
<update id="updateByBatchPrimaryKey" parameterType="java.util.Map"> UPDATE t_goods SET NODE_ID = #{nodeId} WHERE GOODS_ID IN (${goodsIdList}) </update>
实际的sql
UPDATE t_goods SET NODE_ID = ? WHERE GOODS_ID IN (1,2,5);
2.传map/传list
NODE_ID从map中取出来,goodsIdList是用list拼接出来的
<update id="updateByBatchPrimaryKey" parameterType="java.util.Map"> UPDATE t_goods SET NODE_ID = #{nodeId} WHERE GOODS_ID IN <foreach collection="list" index="index" item="item" open="(" separator="," close=")"> #{item.goodsId} </foreach></update>
实际的sql
UPDATE t_goods SET NODE_ID = ? WHERE GOODS_ID IN (1,2,5);
参考文章:主力:https://blog.csdn.net/xyjawq1/article/details/74129316/辅助:https://www.jianshu.com/p/041bec8ae6d3
到此这篇关于Mybatis中updateBatch实现批量更新 的文章就介绍到这了,更多相关Mybatis 批量更新内容请搜索盛行IT以前的文章或继续浏览下面的相关文章希望大家以后多多支持盛行IT!
郑重声明:本文由网友发布,不代表盛行IT的观点,版权归原作者所有,仅为传播更多信息之目的,如有侵权请联系,我们将第一时间修改或删除,多谢。