有的数据要实现历史数据的可追溯性,我们使用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);
}
}