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);
    }
}
相关推荐
我是唐青枫5 小时前
Java MyBatis 实战指南:XML 映射、动态 SQL 与数据访问层设计
java·mybatis
碎碎念_4921 天前
SpringBoot和MyBatis框架·速通版
spring boot·后端·mybatis
唐青枫1 天前
Java MyBatis 实战指南:XML 映射、动态 SQL 与数据访问层设计
java·mybatis
一条泥憨鱼1 天前
苍穹外卖【day3|菜品管理】
java·数据库·sql·mysql·mybatis
INFINI Labs2 天前
Elasticsearch 6/7/8 到 Easysearch 2.x 迁移指南
大数据·elasticsearch·mybatis·向量·snapshot
CodeStats2 天前
从JDBC时代到MyBatis封神:SQL全流程手写ORM实战
sql·mysql·mybatis
我登哥MVP2 天前
Spring Boot 从“会用”到“精通”:参数绑定体系全景
java·spring boot·spring·servlet·maven·intellij-idea·mybatis
唐青枫2 天前
Java MyBatis-Plus 实战指南:用 BaseMapper、Wrapper 和分页写好数据层
java·mybatis
我登哥MVP3 天前
Spring Boot 从“会用”到“精通”:Model-Map原理
java·spring boot·后端·spring·servlet·maven·mybatis
Full Stack Developme3 天前
MyBatis-Plus 分页使用详解
mybatis