MyBatisPlus-将字段更新为null

场景

业务中某些需求,需要将某个字段设为null。
但是mybatisplus并不会更新为null的字段。
这是因为mybatisplus的更新策略默认为NOT_EMPTY,源码中可以查看。

package com.baomidou.mybatisplus.annotation;

/**
 * 字段策略枚举类
 * <p>
 * 如果字段是基本数据类型则最终效果等同于 {@link #IGNORED}
 *
 * @author hubin
 * @since 2016-09-09
 */
public enum FieldStrategy {
    /**
     * 忽略判断
     */
    IGNORED,
    /**
     * 非NULL判断
     */
    NOT_NULL,
    /**
     * 非空判断(只对字符串类型字段,其他类型字段依然为非NULL判断)
     */
    NOT_EMPTY,
    /**
     * 默认的,一般只用于注解里
     * <p>1. 在全局里代表 NOT_NULL</p>
     * <p>2. 在注解里代表 跟随全局</p>
     */
    DEFAULT,
    /**
     * 不加入 SQL
     */
    NEVER
}

解决方法1

在字段头上加注释,设定局部更新策略。

    @TableField(updateStrategy = FieldStrategy.IGNORED)
    private String stdRegister;

但该方法有一个问题。如果其他接口使用该类时,也没有传该字段参数,那么其他接口会把原来的字段都更新为null,造成原来数据丢失,所以不推荐用这种方法。

解决方法2

用update+updatewrapper可实现,关键还是wrapper中将字段设置为null,如果wrapper没有set为null,则这种方式还是会失败。

UpdateWrapper<StudentMsg> wrapper = new UpdateWrapper<>();
            wrapper.eq("std_id", std.getStdId());
            wrapper.set("std_register", null);
            return this.update(wrapper);

如果还有其他字段需要更新,update方法还可以传入entity,将需要set为null的字段写在wrapper中。

UpdateWrapper<StudentMsg> wrapper = new UpdateWrapper<>();
            wrapper.eq("std_id", std.getStdId());
            wrapper.set("std_register", null);
            return this.update(std, wrapper);

这种方式实际执行的sql是,wrapper会覆盖前边的entity

UPDATE
    std_mes 
SET
    std_name='阿里',
    std_gender='男',
    std_age=10,
    std_register='浙江杭州',
    std_register=null 
WHERE
    (
        std_id = 0
    );

解决方法3

用mapper写sql实现

<update id="setRegisterNull">
        update std_mes set std_register = null where std_id = #{id};
    </update>
### 解决 MyBatisPlus 更新实体时部分字段设置为 null 的方法 在 MyBatisPlus 中,默认情况下,当执行 `updateById` 方法时,如果某个字段为 `null`,则不会对该字段进行更新[^1]。然而,通过特定配置可以改变这一行为。 #### 使用 updateById 方法强制更新为空 为了确保即使字段为 `null` 也会被写入数据库,可以在调用 `updateById` 方法前,确保待更新的对象中包含了这些应设为 `null` 的字段及其设定为 `null`。例如: ```java @Test public void testUpdateByIdWithNullFields() { User user = new User(); user.setId(2L); user.setPassword(null); // 明确将密码字段置空 int count = userMapper.updateById(user); } ``` 此代码片段展示了如何显式地把 `password` 字段设置成 `null` 并提交给 `updateById` 进行保存[^2]。 #### 配合 Wrapper 实现更精确控制 对于更加复杂的场景,比如只想针对某些条件来决定哪些字段应该被清空,则可借助于 `Wrapper` 来构建动态 SQL 查询语句。下面是一个例子说明怎样仅当满足一定条件下才允许某列变为 NULL: ```java @Test public void testConditionalUpdateToNull(){ UpdateWrapper<User> wrapper = new UpdateWrapper<>(); wrapper.eq("id", 2).set("password", null); int result = userMapper.update(null,wrapper); } ``` 这里创建了一个 `UpdateWrapper` 对象并指定了要匹配记录 ID 和需修改的具体字段名以及新 (即 `null`) 。最后调用了带有两个参数版本的 `update` 函数完成实际的数据变更操作[^3]。 #### 注意事项 - **确认表结构设计合理性**: 如果频繁遇到需要手动处理大量字段为 `null` 的情况,建议重新审视当前数据库的设计是否合理。 - **考虑业务逻辑影响**: 将现有数据项重置为 `null` 可能会对应用程序其他地方造成意想不到的影响,请谨慎评估后再做决策。 - **测试覆盖率保障**: 修改涉及核心功能的部分务必增加足够的单元测试案例覆盖各种边界状况下的表现形式。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值