当前位置: 首页 > news >正文

中州养老项目:Mybatis自动填充拦截器

功能:在新增护理项目的时候,创建人,创建时间和修改时间字段会自动拦截填充,这些公共字段可以省去我们一个一个处理的麻烦

依靠:AutoFillInterceptor拦截器,MybatisConfig配置类

第一步:我们需要借助一个MybatisConfig,@configuration标志着这是一个配置类,我们需要将autoFillInterceptor放在配置类中并用@Bean注解将这个配置类交给spring容器gu

第二步:了解配置类的原理,也就是AutoFillInterceptor,这个拦截器的代码大部分是固定的我用只需要了解它的逻辑,在需要的时候复制代码并根据需求修改

package com.zzyl.intercept;// 引入工具类和依赖
import cn.hutool.core.util.ObjectUtil;
import com.zzyl.utils.EmptyUtil;
import com.zzyl.utils.UserThreadLocal;
import org.apache.ibatis.binding.MapperMethod;
import org.apache.ibatis.executor.Executor;
import org.apache.ibatis.mapping.MappedStatement;
import org.apache.ibatis.mapping.SqlCommandType;
import org.apache.ibatis.plugin.*;
import org.apache.ibatis.reflection.MetaObject;
import org.apache.ibatis.reflection.SystemMetaObject;
import org.springframework.stereotype.Component;import java.time.LocalDateTime;
import java.util.ArrayList;
import java.util.Properties;// 声明这是一个MyBatis拦截器,拦截Executor的update方法
@Intercepts({@Signature(type = Executor.class, method = "update", args = {MappedStatement.class, Object.class})
})
@Component // 声明为Spring组件
public class AutoFillInterceptor implements Interceptor {// 常量定义:字段名private static final String CREATE_BY = "createBy";   // 创建人字段名private static final String UPDATE_BY = "updateBy";   // 更新人字段名private static final String CREATE_TIME = "createTime"; // 创建时间字段名private static final String UPDATE_TIME = "updateTime"; // 更新时间字段名// 拦截逻辑核心方法@Overridepublic Object intercept(Invocation invocation) throws Throwable {// 获取方法参数:MappedStatement和SQL参数对象Object[] args = invocation.getArgs();MappedStatement ms = (MappedStatement) args[0];  // 获取SQL映射信息SqlCommandType sqlCommandType = ms.getSqlCommandType(); // 获取SQL操作类型(INSERT/UPDATE等)Object parameter = args[1];  // 获取SQL参数对象// 仅当参数和SQL类型有效时处理if (parameter != null && sqlCommandType != null) {Long userId = loadUserId();  // 获取当前用户ID(从ThreadLocal)// 处理INSERT操作if (SqlCommandType.INSERT.equals(sqlCommandType)) {// 批量插入场景(参数是ParamMap且包含list字段)if (parameter instanceof MapperMethod.ParamMap) {MapperMethod.ParamMap paramMap = (MapperMethod.ParamMap) parameter;ArrayList list = (ArrayList) paramMap.get("list"); // 获取批量数据列表list.forEach(v -> {// 为每条数据设置创建人、创建时间、更新时间setFieldValByName(CREATE_BY, userId, v);setFieldValByName(CREATE_TIME, LocalDateTime.now(), v);setFieldValByName(UPDATE_TIME, LocalDateTime.now(), v);});paramMap.put("list", list); // 更新修改后的参数} else {// 单条插入场景setFieldValByName(CREATE_BY, userId, parameter);setFieldValByName(CREATE_TIME, LocalDateTime.now(), parameter);setFieldValByName(UPDATE_TIME, LocalDateTime.now(), parameter);}} // 处理UPDATE操作else if (SqlCommandType.UPDATE.equals(sqlCommandType)) {// 设置更新人和更新时间setFieldValByName(UPDATE_BY, userId, parameter);setFieldValByName(UPDATE_TIME, LocalDateTime.now(), parameter);}}// 继续执行原始SQL操作return invocation.proceed();}/*** 通过反射设置实体对象的字段值* @param fieldName  字段名(如createBy)* @param fieldVal   字段值(如用户ID)* @param parameter  实体对象*/private void setFieldValByName(String fieldName, Object fieldVal, Object parameter) {// 使用MyBatis的MetaObject操作对象属性MetaObject metaObject = SystemMetaObject.forObject(parameter);// 如果是createBy字段且已有值,则跳过(避免覆盖)if (fieldName.equals(CREATE_BY)) {Object value = metaObject.getValue(fieldName);if (ObjectUtil.isNotEmpty(value)) {return;}}// 如果字段存在setter方法,则设置值if (metaObject.hasSetter(fieldName)) {metaObject.setValue(fieldName, fieldVal);}}// 包装目标对象(Executor),返回代理对象@Overridepublic Object plugin(Object target) {if (target instanceof Executor) {return Plugin.wrap(target, this); // MyBatis提供的包装方法}return target; // 非Executor类型直接返回}// 可读取配置文件属性(本例未使用)@Overridepublic void setProperties(Properties properties) {}/*** 获取当前用户ID(优先级:ThreadLocal用户ID → 管理用户ID → 默认值1)*/public static Long loadUserId() {Long userId = UserThreadLocal.getUserId(); // 从业务线程获取if (ObjectUtil.isNotEmpty(userId)) {return userId;}userId = UserThreadLocal.getMgtUserId(); // 从管理线程获取if (!EmptyUtil.isNullOrEmpty(userId)) {return userId;}return 1L; // 默认值}
}

http://www.dtcms.com/a/312167.html

相关文章:

  • 机器学习项目从零到一:加州房价预测模型(PART 2)
  • 李宏毅深度学习教程 第6-7章 自注意力机制 + Transformer
  • NVIDIA GPU架构
  • 浅拷贝与深拷贝的区别
  • 断路器瞬时跳闸曲线数据获取方式
  • 关于Sort的补充
  • SpringBoot 02 AOP
  • 王者荣耀模拟器:一款基于Python的文本角色扮演游戏
  • 译| Netflix内容推荐模型的一些改进方向
  • 测试时扩散的深度研究助手
  • Redis实战(6)-- 慢查询运用与理解
  • 图像加密学习日志————论文学习DAY4
  • 档案馆可视化:历史宝库的数字新貌
  • 使用 MySQL Shell 进行 MySQL 单机到 InnoDB Cluster 的数据迁移实践
  • 【Django】-5- ORM的其他用法
  • RAWINPUT避坑指南(涉及GetRawInputData/GetRawInputBuffer)
  • 详解Python标准库之命令行界面库
  • .env 文件
  • WinMerge:文件对比工具,支持各种格式文件对比,永久免费使用!
  • 软件交付终极闸口:验收测试全解析
  • GCC(GNU Compiler Collection)与人工智能实例
  • 基于深度学习的医学图像分析:使用MobileNet实现医学图像分类
  • 第14届蓝桥杯Python青少组中/高级组选拔赛(STEMA)2022年12月18日真题
  • 【Linux】Linux编译器-gcc/g++使用
  • 广东省省考备考(第六十四天8.2)——判断推理(重点回顾)
  • 【领域热点】人工智能与大模型应用:开启软件领域新革命
  • 无法连接到 Google Container Registry (GCR)** 导致的镜像拉取失败
  • 数据结构与算法:字符串哈希
  • C# 中抽象类、密封类、静态类和接口的区别
  • 数据处理和统计分析——09 数据分组