【Mybatis-Plus 扩展功能】自动填充功能接口 MetaObjectHandler

前言

MetaObjectHandler用在数据库持久化之前,用来对实体对象的公共字段进行统一赋值操作。例如常用来赋值创建时间、创建人、更新时间、更新人等字段
注意:本文使用的是3.0.7.1版本的mybatis-plus(旧项目原因),相对于现在的[v3.5.3] 2022.12.29版本较为老旧,对比官方文档使用

一、MetaObjectHandler 接口

官方解释:官方链接MetaObjectHandler接口

  • 值填充失效的情况(有三种):
    ①被填充的entity对象的字段值不为null;如,要给creator填充,entity为{creator: ‘小明’},则失效;
    ②要填充到entity对象的值为null;如,要给creator填充,但填充的creatorFill(非实体)值为null,则不失效
    ③在update(T t,Wrapper updateWrapper)方法中t为null时,自动填充也会失效

  • 值填充生效的情况: 除了值不填充之外的所有情况(手动狗头)

二、使用

第一步:自定义实现元对象处理器MetaObjectHandler接口,重写插入和更新方法insertFillupdateFill

@Component
public class DateFillHandler implements MetaObjectHandler {

    @Autowired
    @Lazy //(生产过程中这里出现了循环依赖报错,然后加了延迟加载解决,没这个问题的可以跳过)
    private UserService userService;

    @Override
    public void insertFill(MetaObject metaObject) {
        User userInfo = userService.getCurrentUser();
        if (userInfo.getUsername() != null){
        	// 对新增时使用了createBy、updateBy标签的字段赋予用户名(与字段同名最好,不用再配)
        	// createTime和updateTime同上
            this.setFieldValByName("createBy",userInfo.getUsername(),metaObject);
            this.setFieldValByName("updateBy",userInfo.getUsername(),metaObject);
        }
        this.setFieldValByName("createTime", new Timestamp(System.currentTimeMillis()), metaObject);
        this.setFieldValByName("updateTime", new Timestamp(System.currentTimeMillis()), metaObject);
    }

    @Override
    public void updateFill(MetaObject metaObject) {
        User userInfo = userService.getCurrentUser();
         // 对新增时使用了updateBy、updateTime标签的字段赋予用户名
        if (userInfo.getUsername() != null){
            this.setFieldValByName("updateBy",userInfo.getUsername(),metaObject);
        }
        this.setFieldValByName("updateTime", new Timestamp(System.currentTimeMillis()), metaObject);
    }
}

第二步:使用@TableField注解,对新增使用FieldFill.INSERT,对更新使用FieldFill.UPDATE,插入和更新都赋值使用FieldFill.INSERT_UPDATE

public class User implements Serializable {
    private static final long serialVersionUID = 1L;

	/**主键*/
    private String id;
    
	/**创建人  插入时填充*/
    @TableField(fill = FieldFill.INSERT)
    private String createBy;
    
	/**创建日期 插入时填充*/
    @TableField(fill = FieldFill.INSERT)
    private Date createTime;
    
	/**更新人 插入和更新时都填充*/
    @TableField(fill = FieldFill.INSERT_UPDATE)
    private String updateBy;
    
	/**更新日期 插入和更新时都填充*/
    @TableField(fill = FieldFill.INSERT_UPDATE)
    private Date updateTime;
    /**
    也可只是更新时填充
    @TableField(fill = FieldFill.UPDATE)
    private Date updateTime;
	*/
}

三、枚举说明

public enum FieldFill {
    /**
     * 默认不处理
     */
    DEFAULT,
    /**
     * 插入填充字段
     */
    INSERT,
    /**
     * 更新填充字段
     */
    UPDATE,
    /**
     * 插入和更新填充字段
     */
    INSERT_UPDATE
}

四、update(T t,Wrapper updateWrapper)失效举例

失效情况:不指定更新对象时无效

LambdaUpdateWrapper<CarInformation> updateWrapper = new LambdaUpdateWrapper();
//此处省略条件和set值代码
userXxxMapper.update(null, updateWrapper)

有效情况:指定更新对象之后填充值有效

LambdaUpdateWrapper<UserXxx> updateWrapper = new LambdaUpdateWrapper();
UserXxx userXxx = new UserXxx("小明");
//此处省略条件和set值代码
userXxxMapper.update(userXxx, updateWrapper)

五、官方示例

新版本使用strictInsertFill()strictUpdateFill方法。还有fillStrategy方法,根据自己的版本进行选择

@Slf4j
@Component
public class MyMetaObjectHandler implements MetaObjectHandler {

    @Override
    public void insertFill(MetaObject metaObject) {
        log.info("start insert fill ....");
        this.strictInsertFill(metaObject, "createTime", LocalDateTime.class, LocalDateTime.now()); // 起始版本 3.3.0(推荐使用)
        // 或者
        this.strictInsertFill(metaObject, "createTime", () -> LocalDateTime.now(), LocalDateTime.class); // 起始版本 3.3.3(推荐)
        // 或者
        this.fillStrategy(metaObject, "createTime", LocalDateTime.now()); // 也可以使用(3.3.0 该方法有bug)
    }

    @Override
    public void updateFill(MetaObject metaObject) {
        log.info("start update fill ....");
        this.strictUpdateFill(metaObject, "updateTime", LocalDateTime.class, LocalDateTime.now()); // 起始版本 3.3.0(推荐)
        // 或者
        this.strictUpdateFill(metaObject, "updateTime", () -> LocalDateTime.now(), LocalDateTime.class); // 起始版本 3.3.3(推荐)
        // 或者
        this.fillStrategy(metaObject, "updateTime", LocalDateTime.now()); // 也可以使用(3.3.0 该方法有bug)
    }
}
  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
MyBatis-plus是一个优秀的持久层框架,它在MyBatis的基础上进行了扩展和增强。MetaObjectHandlerMyBatis-plus提供的一个接口,用于自动填充实体类中的字段值。但有时候我们可能会遇到MetaObjectHandler不生效的问题。 出现MetaObjectHandler不生效的问题可能有以下几个原因: 1. 没有正确配置MetaObjectHandler:在使用MyBatis-plus时,需要在配置文件中配置MetaObjectHandler的实现类,并将其注入到SqlSessionFactory中。确保配置正确,使得MyBatis-plus能够正确地识别和使用MetaObjectHandler。 2. 实体类没有使用@TableField注解:在需要自动填充的字段上,需要使用@TableField注解进行标记。确保实体类中需要自动填充的字段都使用了@TableField注,并设置了相应的属性值。 3. 字段名与列名不一致:如果实体类中的字段名与数据库表中的列名不一致,需要使用@TableField注解value属性指定数据库列名。确保字段名与数据库列名一致,以便MyBatis-plus能够正确地进行自动填充。 4. 数据库表没有设置默认值或触发器:如果需要自动填充的字段在数据库表中没有设置默认值或触发器,那么在插入或更新数据时,MetaObjectHandler也无法生效。确保数据库表中需要自动填充的字段设置了默认值或触发器。 如果以上几点都没有问题,但仍然遇到MetaObjectHandler不生效的问题,可以尝试检查MyBatis-plus的版本是否与其他依赖库存在冲突,或者查看日志文件以获取更多的错误信息。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值