MyBatis-Plus更新字段为null时,update语句为null字段不生效的解决方法

1.异常说明:

mapper.updateById()时, set为null 未生效,其他字段更新。


   puDeclareElements.setStatus(OrderEnum.DRAFTS.getType());
   puDeclareElements.setSubmitBy(null);
   puDeclareElementsService.updateById(puDeclareElements)
   

2.原理:

MyBatis-Plus对字段的验证策略导致的,MyBatis-Plus默认进了不是全量更新的策略,默认忽略为null 的字段的

3.解决办法:

1)修改MyBatis-Plus 全局默认策略

缺点:这样做会对所有的字段都忽略判断,如果一些字段不想要修改,但是传值的时候没有传递过来,就会被更新为null


mybatis-plus:
  global-config:
  	#字段策略 0:"忽略判断",1:"非 NULL 判断",2:"非空判断"
    field-strategy: 0

2)修改实体类注解,改变字段的忽略判断

缺点:需要注意数据库字段是否设置为 非null ,如果为非null 也更新不成功


@TableField( updateStrategy = FieldStrategy.IGNORED)
private Date settleTime;

4.字段策略全解

1)官方文档

Mybatis-Plus中FieldStrategy说明:​ ​https://baomidou.com/pages/223848/#tableid​

2)字段策略介绍​​

1、FieldStrategy作用

Mybatis-Plus字段策略FieldStrategy的作用主要是在进行新增、更新时,根据配置的策略判断是否对实体对象的值进行空值判断,如果策略为字段不能为空,则不会对为空的字段进行赋值或更新。
同样,在进行where条件查询时,根据whereStrategy策略判断是否对字段进行空值判断,如果策略为字段不能为空,则为空的字段不会作为查询条件组装到where条件中。

三个配置,对应三种使用场景

​​insertStrategy​​ : 在insert操作时的字段策略,是否进行空值判断,插入空值
​​updateStrategy : 在update操作时的字段策略,是否进行空值判断,插入空值
​​whereStrategy : ​​在where条件组装时,是否进行控制判断,将空值作为查询条件

在这里插入图片描述

2、FieldStrategy类型

FieldStrategy的源码中,一共有5种策略类型

public enum FieldStrategy {
    IGNORED,
    NOT_NULL,
    NOT_EMPTY,
    DEFAULT,
    NEVER;

    private FieldStrategy() {
    }
}

每种策略的作用

描述
IGNORED

忽略空值判断,实体对象的字段是什么值就用什么值更新,支持null值更新操作

NOT_NULL

进行非NULL判断,也是默认策略,相当于age!=null

NOT_EMPTY

进行非空判断,主要是针对字符串类型,相当于name != null and name != ‘’

NEVER

从不更新,不管字段是否有值,都不进行更新

DEFAULT

追随全局配置

3、FieldStrategy配置

① 全局策略配置

在全局配置中,三者的默认值都是​​FieldStrategy.NOT_NULL​​,即进行空值判断,不对NULL值数据进行处理。

public DbConfig() {
    this.idType = IdType.ASSIGN_ID;
    this.tableUnderline = true;
    this.capitalMode = false;
    this.logicDeleteValue = "1";
    this.logicNotDeleteValue = "0";
    this.insertStrategy = FieldStrategy.NOT_NULL;
    this.updateStrategy = FieldStrategy.NOT_NULL;
    this.whereStrategy = FieldStrategy.NOT_NULL;
}

在spring boot中可以通过配置属性修改全局字段策略:

mybatis-plus.global-config.db-config.update-strategy=not_null
mybatis-plus.global-config.db-config.insert-strategy=not_null
mybatis-plus.global-config.db-config.where-strategy=not_null
② 单字段策略配置

在实体对象中,通过​​@TableField​​注解可以针对单个字段指定字段策略。

3)实战说明

1、忽略判断-IGNORED

@TableName(value ="user")
@Data
public class User implements Serializable {
   @TableId
   private Long id;
   private String name;
   private Integer age;
   @TableField(updateStrategy = FieldStrategy.IGNORED)
   private String email;
}

再次执行上面的单元测试:

在这里插入图片描述

PS:后面就不一一举例了。可以看官网。写的很详细。

MyBatis Plus 的批量更新操作中,如果某个字段为空值,在某些场景下可能会导致该字段未被成功更新的问题。这通常是因为 MyBatis Plus 默认会忽略掉 `null` 值的字段。 ### 分析原因及解决方案 #### 可能的原因 1. **策略问题**:MyBatis Plus 中,默认开启的是 `update selective` 策略,即只对非空字段生成 SQL 更新语句。 2. **FieldFill 注解影响**:如果你给字段加了 `@TableField(fill = FieldFill.INSERT)` 或者类似注解,并且当前不在指定的填充范围,则可能不会生效。 3. **SQL 生成机制**:当字段为 `null` MyBatis Plus 不会在生成的动态 SQL 中包含此字段--- ### 解决方案 以下是几种常见的解决办法: #### 方法一:修改全局配置 可以在 `mybatis-plus.global-config.db-config.id-type` 配置文件中调整默认的行为模式。例如: ```yaml mybatis-plus: global-config: db-config: field-strategy: NOT_NULL # 设置为NOT_NULL允许插入NULL值 ``` #### 方法二:手动设置 updateWrapper 条件 通过构造自定义的 `UpdateWrapper` 明确地添加需要更新的所有字段条件,包括那些可能是空值的情况: ```java UpdateWrapper<YourEntity> wrapper = new UpdateWrapper<>(); wrapper.eq("id", yourId); yourEntity.setYourField(null); // 直接将目标字段设为 null 并传入 entity 对象内 int result = yourMapper.update(yourEntity, wrapper); if (result > 0) { System.out.println("更新成功"); } else { System.out.println("无符合条件的数据记录或失败!"); } ``` 这种方法可以强制把字段纳入更新范围内,即便它的新值是 `null`。 #### 方法三:使用 @TableField(strategy=FieldStrategy.NOT_NULL) 针对个别实体类属性单独处理,标记其支持 `null` 插入/更新: ```java @TableField(updateStrategy = FieldStrategy.IGNORED, insertStrategy = FieldStrategy.NOT_NULL) private String yourFieldName; ``` 注意这里的选项含义: - `IGNORED`: 完全忽视本字段是否参与任何 DML 操作; - `NOT_NULL`: 当前字段若存在不是 NULL 则加入操作列表;反之跳过。 --- ### 示例代码片段 下面是一段完整演示如何完成带 `null` 字段更新的例子: ```java // 构建待更新的对象实例 User user = new User(); user.setId(1L); // 主键ID用于定位数据行 user.setName(null); // 我们希望这一列为 null更新进去 user.setEmail("newemail@example.com"); // 创建包装器设定过滤规则 UpdateWrapper<User> uw = new UpdateWrapper<>(); uw.eq("id", user.getId()); // 调用 mapper 层执行更新动作 boolean success = userService.update(user, uw); System.out.println(success ? "用户信息已更新" : "未能找到对应条目进行更改"); ``` ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值