MyBatis Plus 中 update_time 字段自动填充失效的原因分析及解决方案(最新整理)

 更新时间:2025年06月11日 15:17:56   作者:代码怪兽大作战  
在使用MyBatis Plus时,通常我们会在数据库表中设置create_time和update_time两个字段,借助 MyBatis Plus 的自动填充功能来维护这些时间字段,下面通过本文介绍MyBatis Plus 中 update_time 字段自动填充失效的原因分析及解决方案,感兴趣的朋友一起看看吧

前言

在使用 MyBatis Plus 时,通常我们会在数据库表中设置 create_timeupdate_time 两个字段,借助 MyBatis Plus 的 自动填充功能 来维护这些时间字段。

但是,有时候你可能会遇到 update_time 字段未自动更新的情况,哪怕代码中已经配置了 MetaObjectHandler。本文将针对这一问题,分析可能的原因,并提供详细的解决方案。

一、问题现象

假设我们有如下的 updateFill() 方法:

@Override
public void updateFill(MetaObject metaObject) {
	log.info("执行更新填充...");
    this.strictInsertFill(metaObject, "updateTime", Timestamp.class, Timestamp.valueOf(LocalDateTime.now())); 
    this.strictInsertFill(metaObject, "updateBy", BigInteger.class, userId); 
}

但是执行更新操作后:

  • 数据库中的 update_time 字段未更新
  • 没有报错,但 updateTime 依然是旧值
  • updateBy 可能也未更新

接下来我们来分析可能的原因。

二、原因分析

1. 使用了 strictInsertFill/strictUpdateFill 导致更新失效

原因:

strictInsertFill/strictUpdateFill 主要用于插入数据时的自动填充。即使在 updateFill() 中使用它,也不会生效。

原理解释:

strictInsertFill/strictUpdateFill 的作用是 如果字段已有值则不会覆盖,而在更新场景中,一般实体对象中 update_time 已有值。

解决方案:

strictInsertFill/strictUpdateFill 替换为 setFieldValByName

@Override
public void updateFill(MetaObject metaObject) {
    log.info("start update fill ....");
    this.setFieldValByName("updateTime", Timestamp.valueOf(LocalDateTime.now()), metaObject);
    this.setFieldValByName("updateBy", userId, metaObject);
}

原因分析补充:

  • strictInsertFill/strictUpdateFill插入/更新数据时填充,不适用于更新场景。
  • setFieldValByName → 更新数据时直接覆盖字段值,即使原值存在。

2. 实体类注解配置错误

原因:

实体类中的字段没有正确标注 @TableField(fill = FieldFill.INSERT_UPDATE),导致自动填充机制无法生效。

解决方案:

在实体类中正确设置注解:

import com.baomidou.mybatisplus.annotation.FieldFill;
import com.baomidou.mybatisplus.annotation.TableField;
import lombok.Data;
import java.time.LocalDateTime;
@Data
public class User {
    private Long id;
    private String name;
    @TableField(fill = FieldFill.INSERT)
    private LocalDateTime createTime;
    @TableField(fill = FieldFill.INSERT_UPDATE)
    private LocalDateTime updateTime;
    @TableField(fill = FieldFill.INSERT_UPDATE)
    private BigInteger updateBy;
}

注意:

  • FieldFill.INSERT_UPDATE:用于插入和更新时自动填充。
  • 如果没有加上这个注解,MyBatis Plus 不会执行自动填充逻辑。

3. MetaObjectHandler 未生效

原因:

MetaObjectHandler 需要被 Spring 管理。如果没有加上 @Component 注解或没有被扫描到,自动填充方法不会执行。

解决方案:

确保 FieldAutoFillHandler 类上有 @Component 注解:

import com.baomidou.mybatisplus.core.handlers.MetaObjectHandler;
import lombok.extern.slf4j.Slf4j;
import org.apache.ibatis.reflection.MetaObject;
import org.springframework.stereotype.Component;
@Slf4j
@Component
public class FieldAutoFillHandler implements MetaObjectHandler {
    @Override
    public void insertFill(MetaObject metaObject) {
        log.info("执行插入填充...");
        this.setFieldValByName("createTime", LocalDateTime.now(), metaObject);
        this.setFieldValByName("updateTime", LocalDateTime.now(), metaObject);
    }
    @Override
    public void updateFill(MetaObject metaObject) {
        log.info("执行更新填充...");
        this.setFieldValByName("updateTime", LocalDateTime.now(), metaObject);
    }
}

注意:

如果项目中有多个 MetaObjectHandler,需要确保不会有冲突。

4. 使用自定义 SQL 导致自动填充失效

原因:

如果你在 Mapper.xml 中自定义了 update 语句,那么 MyBatis Plus 不会触发自动填充逻辑。

解决方案:

方法 1:在 XML 中手动设置 update_time 字段:

<update id="updateUser" parameterType="com.example.entity.User">
    UPDATE user
    SET name = #{name},
        update_time = NOW(),
        update_by = #{updateBy}
    WHERE id = #{id}
</update>

方法 2:直接使用 BaseMapper 提供的 updateById() 等方法,MyBatis Plus 会自动触发填充:

User user = userMapper.selectById(1L);
user.setName("李四");
userMapper.updateById(user);

5. 字段类型不匹配

原因:

数据库中 update_time 的数据类型与 Java 实体类的类型不匹配,导致填充失败。

解决方案:

确保数据库字段类型和实体类一致:

ALTER TABLE user
MODIFY update_time TIMESTAMP NULL DEFAULT NULL;
private LocalDateTime updateTime;

注意:

  • 使用 LocalDateTime 处理时间字段是最佳实践。
  • 如果数据库字段为 datetimetimestamp,不要使用 String 类型。

三、总结:常见原因与解决方法对照表

原因现象解决方法
使用了 strictInsertFillupdate_time 未更新使用 setFieldValByName 替代
实体类注解错误自动填充无效确保字段上加了 @TableField(fill = FieldFill.INSERT_UPDATE)
MetaObjectHandler 未生效updateFill() 不执行确保加上 @Component 注解
使用自定义 SQL 覆盖了默认方法填充逻辑未触发使用 BaseMapper.updateById() 或在 SQL 中手动填充
数据类型不匹配字段更新失败数据库字段使用 timestamp 且实体类使用 LocalDateTime

四、推荐写法

最终推荐的 updateFill() 写法如下:

@Override
public void updateFill(MetaObject metaObject) {
    log.info("执行更新填充...");
    this.setFieldValByName("updateTime", LocalDateTime.now(), metaObject);
    this.setFieldValByName("updateBy", userId, metaObject);
}

通过这种方式,确保 update_time 字段在更新时自动更新,同时避免了不必要的填充失效问题。

到此这篇关于MyBatis Plus 中 update_time 字段自动填充失效的原因分析及解决方案(最新整理)的文章就介绍到这了,更多相关MyBatis Plus update_time 自动填充失效内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • SpringBoot使用拦截器Interceptor实现统一角色权限校验

    SpringBoot使用拦截器Interceptor实现统一角色权限校验

    角色权限校验,是保证接口安全必备的能力:有权限才可以操作,所以,一般对于这种通用逻辑,推荐不与主业务逻辑耦合,那么怎么来解耦,那么本文小编就给大家详细讲解如何使用拦截器Interceptor实现统一角色权限校验,需要的朋友可以参考下
    2023-07-07
  • 在IntelliJ IDEA中高效运行与调试Spring Boot项目的实战步骤

    在IntelliJ IDEA中高效运行与调试Spring Boot项目的实战步骤

    本章详解SpringBoot项目导入IntelliJ IDEA的流程,教授运行与调试技巧,包括断点设置与变量查看,奠定高效开发基础,本文给大家介绍在IntelliJ IDEA中高效运行与调试Spring Boot项目的实战步骤,感兴趣的朋友一起看看吧
    2025-07-07
  • Java中 equals 重写时为什么一定也要重写 hashCode

    Java中 equals 重写时为什么一定也要重写 hashCode

    这篇文章主要介绍了Java中 equals 重写时为什么一定也要重写 hashCode,equals 方法和 hashCode 方法是 Object 类中的两个基础方法,它们共同协作来判断两个对象是否相等,所以之间到底有什么联系呢,接下来和小编一起进入文章学习该内容吧
    2022-05-05
  • Spring Boot使用过滤器和拦截器分别实现REST接口简易安全认证示例代码详解

    Spring Boot使用过滤器和拦截器分别实现REST接口简易安全认证示例代码详解

    这篇文章主要介绍了Spring Boot使用过滤器和拦截器分别实现REST接口简易安全认证示例代码,通过开发实践,理解过滤器和拦截器的工作原理,需要的朋友可以参考下
    2018-06-06
  • java 优雅关闭线程池的方案

    java 优雅关闭线程池的方案

    这篇文章主要介绍了java 优雅关闭线程池的方案,帮助大家更好的理解和学习Java,感兴趣的朋友可以了解下
    2020-11-11
  • SpringBoot中的ThreadLocal保存请求用户信息的实例demo

    SpringBoot中的ThreadLocal保存请求用户信息的实例demo

    线程局部变量,创建一个线程变量后,针对这个变量可以让每个线程拥有自己的变量副本,每个线程是访问的自己的副本,与其他线程的相互独立,本文介绍SpringBoot中的ThreadLocal保存请求用户信息,需要的朋友可以参考下
    2024-05-05
  • Spark MLlib随机梯度下降法概述与实例

    Spark MLlib随机梯度下降法概述与实例

    这篇文章主要为大家详细介绍了Spark MLlib随机梯度下降法概述与实例,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-08-08
  • Java序列化原理详解

    Java序列化原理详解

    这篇文章主要介绍了Java序列化原理详解,文章围绕主题展开详细的内容介绍,具有一定的参考价值,感兴趣的小伙伴可以参考一下
    2022-06-06
  • Spring实战之使用ClassPathResource加载xml资源示例

    Spring实战之使用ClassPathResource加载xml资源示例

    这篇文章主要介绍了Spring实战之使用ClassPathResource加载xml资源,结合实例形式分析了Spring使用ClassPathResource加载xml资源的具体实现步骤与相关操作技巧,需要的朋友可以参考下
    2019-12-12
  • java 获取内存使用率的流程实例详解

    java 获取内存使用率的流程实例详解

    这篇文章主要为大家介绍了java 获取内存使用率的流程实例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-12-12

最新评论