mybatis-plus批量更新updateBatchById问题

2023-08-06 0 4,291

目录

前言

在使用mybatis-plus过程中,有很多插件都特别优秀,不仅使我们代码更加优雅,也提升了效率。

其中有个批量插入的插件insertBatchSomeColumn使用起来也挺方便的,但是批量更新一直没有官方插件,网络上面也没有找到靠谱的,于是就参照mybatis-plus这些官方的方法自定义了一个批量更新的方法。

实现效果

案例:用户排序

mybatis-plus批量更新updateBatchById问题

最终更新语句:

UPDATE sys_user 
SET user_order =
CASE
		id 
		WHEN 1 THEN	1 
		WHEN 2 THEN	2 
		WHEN 3 THEN	3 
		WHEN 4 THEN	4 
END 
WHERE tenant_id = 1 
AND id IN (1,2,3,4)

批量新增插件的配置

定义一个自己的BaseMapper继承自mybatis-plus的BaseMapper,声明批量新增方法,如下:

public interface MyBaseMapper<T> extends BaseMapper<T> {
    /**
     * 批量插入
     *
     * @param entityList 实体列表
     * @return 影响行数
     */
    int insertBatchSomeColumn(Collection<T> entityList);
}

把批量新增方法添加到方法列表中:

/**
 * 通用方法注入
 */
public class MySqlInjector extends DefaultSqlInjector {
    @Override
    public List<AbstractMethod> getMethodList(Class<?> mapperClass) {
        List<AbstractMethod> methodList = super.getMethodList(mapperClass);
        // 添加批量新增方法
        methodList.add(new InsertBatchSomeColumn());
        return methodList;
    }
}

MybatisPlusConfig配置中注入bean:

    @Bean
    public MySqlInjector mySqlInjector() {
        return new MySqlInjector();
    }

业务Mapper继承自自定义的MyBaseMapper,则就可以使用批量新增方法了。

下面进入正题

updateBatchById实现

自定义方法枚举

参照官方的SqlMethod,创建枚举MySqlMethod,并定义批量更新方法,如下:

public enum MySqlMethod {
    UPDATE_BATCH_BY_ID(\"updateBatchById\", \"通过主键批量更新数据\", \"<script>UPDATE %s \\n%s \\nWHERE %s IN %s\\n</script>\");
    private final String method;
    private final String desc;
    private final String sql;
    MySqlMethod(String method, String desc, String sql) {
        this.method = method;
        this.desc = desc;
        this.sql = sql;
    }
    public String getMethod() {
        return this.method;
    }
    public String getDesc() {
        return this.desc;
    }
    public String getSql() {
        return this.sql;
    }
}

自定义批量更新方法

定义UpdateBatchById继承自AbstractMethod,实现其抽象方法injectMappedStatement,功能就是拼接sql,具体实现如下:

/**
 * 通过ID批量更新
 */
public class UpdateBatchById extends AbstractMethod {
    private static final long serialVersionUID = 4198102405483580486L;
    @Override
    public MappedStatement injectMappedStatement(Class<?> mapperClass, Class<?> modelClass, TableInfo tableInfo) {
        MySqlMethod sqlMethod = MySqlMethod.UPDATE_BATCH_BY_ID;
        String sql = String.format(sqlMethod.getSql(), tableInfo.getTableName(), this.sqlSet(tableInfo), tableInfo.getKeyColumn(), this.sqlIn(tableInfo.getKeyProperty()));
        SqlSource sqlSource = this.languageDriver.createSqlSource(this.configuration, sql, modelClass);
        return this.addUpdateMappedStatement(mapperClass, modelClass, sqlMethod.getMethod(), sqlSource);
    }
    private String sqlSet(TableInfo tableInfo) {
        List<TableFieldInfo> fieldList = tableInfo.getFieldList();
        StringBuilder sb = new StringBuilder();
        for (TableFieldInfo fieldInfo : fieldList) {
            sb.append(\"<if test=\\\"ew.updateFields.contains(&quot;\").append(fieldInfo.getColumn()).append(\"&quot;)\\\">\")
                    .append(fieldInfo.getColumn()).append(\" =\\n\")
                    .append(\"CASE \").append(tableInfo.getKeyColumn()).append(\"\\n\")
                    .append(\"<foreach collection=\\\"list\\\" item=\\\"et\\\" >\\n\")
                    .append(\"WHEN #{et.\").append(tableInfo.getKeyProperty()).append(\"} THEN #{et.\").append(fieldInfo.getProperty()).append(\"}\\n\")
                    .append(\"</foreach>\\n\").append(\"END ,\\n\")
                    .append(\"</if>\\n\");
        }
        return \"<set>\\n\" + sb + \"</set>\";
    }
    private String sqlIn(String keyProperty) {
        StringBuilder sb = new StringBuilder();
        sb.append(\"<foreach collection=\\\"list\\\" item=\\\"et\\\" separator=\\\",\\\" open=\\\"(\\\" close=\\\")\\\">\\n\")
                .append(\"#{et.\").append(keyProperty).append(\"}\")
                .append(\"</foreach>\\n\");
        return sb.toString();
    }
}

到了这一步已经能够基本实现功能了,但是无法控制需要更新的字段,继续看下面。

自定义更新wrapper

自定义UpdateBatchWrapper继承自AbstractLambdaWrapper,此类主要为updateFields属性设置值,拼接sql的时候只对设置的属性更新,其他属性不变。

public class UpdateBatchWrapper<T> extends AbstractLambdaWrapper<T, UpdateBatchWrapper<T>> {
    private static final long serialVersionUID = 114684162001472707L;
    /**
     * 需要更新的字段
     */
    private List<String> updateFields = null;
    @Override
    protected UpdateBatchWrapper<T> instance() {
        this.updateFields = new ArrayList<>();
        return this;
    }
	/**
     * 关键代码,为属性设置值
     */
    @SafeVarargs
    public final UpdateBatchWrapper<T> setUpdateFields(SFunction<T, ?>... columns) {
        this.updateFields = Arrays.asList(columnsToString(columns).split(\",\"));
        return this;
    }
    public List<String> getUpdateFields() {
        return updateFields;
    }
}

参照批量新增把方法添加到方法列表

MyBaseMapper增加配置:

 /**
     * 通过ID批量更新数据
     *
     * @param entityList 实体列表
     * @return 影响行数
     */
    int updateBatchById(@Param(\"list\") Collection<T> entityList, @Param(\"ew\") Wrapper<T> updateWrapper);

MySqlInjector增加配置:

// 添加批量更新方法
methodList.add(new UpdateBatchById());

测试updateBatchById

创建一个接口saveUserOrder实现用户排序,进而检查批量更新方法。

controller层

    @PostMapping(\"saveUserOrder\")
    @ApiOperation(\"用户排序\")
    public Result saveUserOrder(@RequestBody List<OrderUserSO> soList) {
        sysService.saveUserOrder(soList);
        return Result.success();
    }

service层

@Override
    public void saveUserOrder(List<OrderUserSO> soList) {
        // 业务实体转换为数据库实体
        List<SysUser> userList = JSONUtil.toList(JSONUtil.toJsonStr(soList), SysUser.class);
        // 批量更新-设置更新字段为userOrder
        sysUserMapper.updateBatchById(userList,
                new UpdateBatchWrapper<SysUser>()
                        .setUpdateFields(SysUser::getUserOrder));
    }

OrderUserSO实体

@Data
@ApiModel(\"用户排序业务实体\")
public class OrderUserSO implements Serializable {
    private static final long serialVersionUID = 509541044282315352L;
    @ApiModelProperty(value = \"用户ID\", required = true)
    @NotNull
    private Integer id;
    @ApiModelProperty(value = \"用户顺序\", required = true)
    @NotNull
    private Integer userOrder;
}

见证奇迹的时刻到了…去文章开头见证奇迹吧。

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持。

资源下载此资源下载价格为1小猪币,终身VIP免费,请先
由于本站资源来源于互联网,以研究交流为目的,所有仅供大家参考、学习,不存在任何商业目的与商业用途,如资源存在BUG以及其他任何问题,请自行解决,本站不提供技术服务! 由于资源为虚拟可复制性,下载后不予退积分和退款,谢谢您的支持!如遇到失效或错误的下载链接请联系客服QQ:442469558

:本文采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可, 转载请附上原文出处链接。
1、本站提供的源码不保证资源的完整性以及安全性,不附带任何技术服务!
2、本站提供的模板、软件工具等其他资源,均不包含技术服务,请大家谅解!
3、本站提供的资源仅供下载者参考学习,请勿用于任何商业用途,请24小时内删除!
4、如需商用,请购买正版,由于未及时购买正版发生的侵权行为,与本站无关。
5、本站部分资源存放于百度网盘或其他网盘中,请提前注册好百度网盘账号,下载安装百度网盘客户端或其他网盘客户端进行下载;
6、本站部分资源文件是经压缩后的,请下载后安装解压软件,推荐使用WinRAR和7-Zip解压软件。
7、如果本站提供的资源侵犯到了您的权益,请邮件联系: 442469558@qq.com 进行处理!

猪小侠源码-最新源码下载平台 Java教程 mybatis-plus批量更新updateBatchById问题 http://www.20zxx.cn/806423/xuexijiaocheng/javajc.html

猪小侠源码,优质资源分享网

常见问题
  • 本站所有资源版权均属于原作者所有,均只能用于参考学习,请勿直接商用。若由于商用引起版权纠纷,一切责任均由使用者承担
查看详情
  • 最常见的情况是下载不完整: 可对比下载完压缩包的与网盘上的容量,建议提前注册好百度网盘账号,使用百度网盘客户端下载
查看详情

相关文章

官方客服团队

为您解决烦忧 - 24小时在线 专业服务