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

本文主要是介绍MyBatis Plus 中 update_time 字段自动填充失效的原因分析及解决方案(最新整理),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

《MyBatisPlus中update_time字段自动填充失效的原因分析及解决方案(最新整理)》在使用MyBatisPlus时,通常我们会在数据库表中设置create_time和update...

前言

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

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

一、问题现象

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

@Override
public void updateFill(MetaObject metaObject) {
	log.info("执行更新填充...");
    this.strictInsertFill(metaObject, "updateTime", Timestamp.clpythonass, 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.TableFieldpython;
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("执行插js入填充...");
        this.setFieldValByName("createTime", LocalDateTime.now(), metaObject);
       China编程 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 自动填充失效内容请搜索编程China编程(www.chinasem.cn)以前的文章或继续浏览下面的相关文章希望大家以后多多支持China编程(www.chinasem.cn)!

这篇关于MyBatis Plus 中 update_time 字段自动填充失效的原因分析及解决方案(最新整理)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


原文地址:http://www.cppcns.com/ruanjian/java/713562.html
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.chinasem.cn/article/1155018

相关文章

MySQL逻辑删除与唯一索引冲突解决方案

《MySQL逻辑删除与唯一索引冲突解决方案》本文探讨MySQL逻辑删除与唯一索引冲突问题,提出四种解决方案:复合索引+时间戳、修改唯一字段、历史表、业务层校验,推荐方案1和方案3,适用于不同场景,感兴... 目录问题背景问题复现解决方案解决方案1.复合唯一索引 + 时间戳删除字段解决方案2:删除后修改唯一字

Python自动化批量重命名与整理文件系统

《Python自动化批量重命名与整理文件系统》这篇文章主要为大家详细介绍了如何使用Python实现一个强大的文件批量重命名与整理工具,帮助开发者自动化这一繁琐过程,有需要的小伙伴可以了解下... 目录简介环境准备项目功能概述代码详细解析1. 导入必要的库2. 配置参数设置3. 创建日志系统4. 安全文件名处

SpringBoot中六种批量更新Mysql的方式效率对比分析

《SpringBoot中六种批量更新Mysql的方式效率对比分析》文章比较了MySQL大数据量批量更新的多种方法,指出REPLACEINTO和ONDUPLICATEKEY效率最高但存在数据风险,MyB... 目录效率比较测试结构数据库初始化测试数据批量修改方案第一种 for第二种 case when第三种

解决1093 - You can‘t specify target table报错问题及原因分析

《解决1093-Youcan‘tspecifytargettable报错问题及原因分析》MySQL1093错误因UPDATE/DELETE语句的FROM子句直接引用目标表或嵌套子查询导致,... 目录报js错原因分析具体原因解决办法方法一:使用临时表方法二:使用JOIN方法三:使用EXISTS示例总结报错原

MyBatis-Plus通用中等、大量数据分批查询和处理方法

《MyBatis-Plus通用中等、大量数据分批查询和处理方法》文章介绍MyBatis-Plus分页查询处理,通过函数式接口与Lambda表达式实现通用逻辑,方法抽象但功能强大,建议扩展分批处理及流式... 目录函数式接口获取分页数据接口数据处理接口通用逻辑工具类使用方法简单查询自定义查询方法总结函数式接口

MySQL 迁移至 Doris 最佳实践方案(最新整理)

《MySQL迁移至Doris最佳实践方案(最新整理)》本文将深入剖析三种经过实践验证的MySQL迁移至Doris的最佳方案,涵盖全量迁移、增量同步、混合迁移以及基于CDC(ChangeData... 目录一、China编程JDBC Catalog 联邦查询方案(适合跨库实时查询)1. 方案概述2. 环境要求3.

SpringSecurity整合redission序列化问题小结(最新整理)

《SpringSecurity整合redission序列化问题小结(最新整理)》文章详解SpringSecurity整合Redisson时的序列化问题,指出需排除官方Jackson依赖,通过自定义反序... 目录1. 前言2. Redission配置2.1 RedissonProperties2.2 Red

MyBatis中$与#的区别解析

《MyBatis中$与#的区别解析》文章浏览阅读314次,点赞4次,收藏6次。MyBatis使用#{}作为参数占位符时,会创建预处理语句(PreparedStatement),并将参数值作为预处理语句... 目录一、介绍二、sql注入风险实例一、介绍#(井号):MyBATis使用#{}作为参数占位符时,会

MySQL 多列 IN 查询之语法、性能与实战技巧(最新整理)

《MySQL多列IN查询之语法、性能与实战技巧(最新整理)》本文详解MySQL多列IN查询,对比传统OR写法,强调其简洁高效,适合批量匹配复合键,通过联合索引、分批次优化提升性能,兼容多种数据库... 目录一、基础语法:多列 IN 的两种写法1. 直接值列表2. 子查询二、对比传统 OR 的写法三、性能分析

MySQL中的LENGTH()函数用法详解与实例分析

《MySQL中的LENGTH()函数用法详解与实例分析》MySQLLENGTH()函数用于计算字符串的字节长度,区别于CHAR_LENGTH()的字符长度,适用于多字节字符集(如UTF-8)的数据验证... 目录1. LENGTH()函数的基本语法2. LENGTH()函数的返回值2.1 示例1:计算字符串