mybatis 批量 update,mybatis foreach批量更新

  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的观点,版权归原作者所有,仅为传播更多信息之目的,如有侵权请联系,我们将第一时间修改或删除,多谢。

留言与评论(共有 条评论)
   
验证码: