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);
    }
}
相关推荐
希望永不加班3 小时前
SpringBoot 自定义 Starter:从零开发一个私有 Starter
java·spring boot·后端·spring·mybatis
未秃头的程序猿7 小时前
💥 MyBatis 面试连环炮:从源码原理到实战避坑,彻底拿下 Offer 通关秘籍
后端·面试·mybatis
A_QXBlms9 小时前
企微群发消息技术实现:定时任务+模板消息
java·mybatis·企业微信
huanmieyaoseng100310 小时前
Mybatis常见面试题
java·开发语言·mybatis
希望永不加班1 天前
SpringBoot 自动配置类加载顺序与优先级
java·spring boot·后端·spring·mybatis
计算机学姐1 天前
基于SpringBoot的房屋交易系统
java·vue.js·spring boot·后端·spring·intellij-idea·mybatis
Rick19931 天前
spring boot和mybatis框架的设计思想和核心逻辑
spring boot·后端·mybatis
weixin_704266051 天前
读取Excel 和 显示预约人数
spring boot·mybatis·excel
计算机学姐1 天前
基于SpringBoot的充电桩预约管理系统【阶梯电费+个性化推荐+数据可视化】
java·vue.js·spring boot·后端·mysql·信息可视化·mybatis