Mybatis中保证时间戳的一致性

有的数据要实现历史数据的可追溯性,我们使用update_time和create_time来记录时间戳,这时数据就要保证时间戳的严格一致性,比如,更新数据时,我们可能会采用先软删除老数据后插入新数据的方式实现,这时候,如果时间有交叉或空隙,可能会导致这个时间点的数据多余或缺少,所以,下面我们就用mybatis-plus框架来实现每次请求共用一个时间戳的方式。

1. 创建一个请求级的时间戳持有者

首先,创建一个组件来存储当前请求的"固定"时间戳。使用 @RequestScope 可以确保每个HTTP请求都拥有独立的实例,不会相互干扰。

java 复制代码
package com.xcmg.finance.tmp.common.mybatis.handler;

import org.springframework.stereotype.Component;
import org.springframework.web.context.annotation.RequestScope;

import java.time.OffsetDateTime;

@Component
@RequestScope
public class RequestTimestampHolder {
    private OffsetDateTime currentTimestamp;

    public OffsetDateTime getCurrentTimestamp() {
        // 如果时间戳不存在,则初始化
        if (currentTimestamp == null) {
            currentTimestamp = OffsetDateTime.now();
        }
        return currentTimestamp;
    }

    // 可选:允许手动重置,用于特殊场景
    public void reset() {
        this.currentTimestamp = null;
    }
}

2. 改造 MetaObjectHandler

接下来,修改 MetaObjectHandler 实现。关键点在于:不再直接调用 LocalDateTime.now(),而是从上面创建的 RequestTimestampHolder 中获取时间戳。

java 复制代码
package com.xcmg.finance.tmp.common.mybatis.handler;

import com.baomidou.mybatisplus.core.handlers.MetaObjectHandler;
import com.xcmg.finance.tmp.common.auth.AuthUtils;
import org.apache.ibatis.reflection.MetaObject;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

import java.time.OffsetDateTime;

@Component
public class XcmgMetaObjectHandler  implements MetaObjectHandler {
    @Autowired
    private RequestTimestampHolder timestampHolder;
    @Override
    public void insertFill(MetaObject metaObject) {
        // 只在插入时填充
        OffsetDateTime ts = timestampHolder.getCurrentTimestamp();
        this.strictInsertFill(metaObject, "createdAt", OffsetDateTime.class, ts);
        this.strictInsertFill(metaObject, "updatedAt", OffsetDateTime.class, ts);
        String username = AuthUtils.getLoginUsername();
        this.strictInsertFill(metaObject, "createdBy", String.class, username);
        this.strictInsertFill(metaObject, "updatedBy", String.class, username);
    }

    @Override
    public void updateFill(MetaObject metaObject) {
        this.setFieldValByName("updatedAt", OffsetDateTime.now(), metaObject);
        this.setFieldValByName("updatedBy", AuthUtils.getLoginUsername(), metaObject);
    }
}
相关推荐
暗暗别做白日梦41 分钟前
MyBatis-Plus 分页查询@Param 注解
mybatis
是梦终空2 天前
计算机源码273—基于SpringBoot+Vue3停车场管理系统带支沙箱支付(源代码+数据库)
数据库·spring boot·vue·mybatis·停车场管理系统·沙箱支付·毕设设计
Yeh2020582 天前
Mybatis笔记一
java·笔记·mybatis
广师大-Wzx2 天前
JavaWeb:后端部分
java·开发语言·spring·servlet·tomcat·maven·mybatis
MacroZheng2 天前
横空出世!IDEA最强MyBatis插件来了,功能很全!
java·后端·mybatis
Java成神之路-2 天前
解析 MyBatis 中 #{} 与 ${}区别及 SQL 注入防范(附 Like/In/Order by 安全写法)
sql·安全·mybatis
Zephyr_02 天前
SQL,MyBatis-Plus,maven,Spring与VUE3
sql·spring·vue·maven·mybatis