公共字段自动填冲代码

java 复制代码
package com.sky.aspect;

import com.sky.annotation.AutoFill;
import com.sky.context.BaseContext;
import com.sky.enumeration.OperationType;
import lombok.extern.slf4j.Slf4j;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.Signature;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.aspectj.lang.reflect.MethodSignature;
import org.springframework.stereotype.Component;

import java.lang.reflect.Method;
import java.time.LocalDateTime;

@Component
@Slf4j
@Aspect
public class AutoFillAspect {
    /**
     * 切入点
     */
    //必须同时满足下面两个条件才可以进行拦截
    @Pointcut("execution(* com.sky.mapper.*.*(..)) && @annotation(com.sky.annotation.AutoFill)")
    public void autoFillPointCut(){}

    @Before("autoFillPointCut()")
    public void autoFill(JoinPoint joinPoint){
        log.info("开始进行公共字段的填充...");

        //1.获取到当前被拦截的方法上的数据库操作类型,也就是说是添加操作呢还是修改操作。
        MethodSignature signature = (MethodSignature) joinPoint.getSignature();//方法签名对象
        AutoFill autoFill = signature.getMethod().getAnnotation(AutoFill.class);//获得方法上的注解对象
        OperationType operationType = autoFill.value();//获得数据库操作类型


        //2.获取到当前被拦截的方法的参数----实体对象

        Object[] args = joinPoint.getArgs();//取回方法中的所有参数,那么我们默认把实体对象放在第一个位置。
        if(args == null || args.length == 0){
            return;
        }
        Object entity = args[0];//为什么用Object接收,因为实体对象不止一种类型。
        
        //3.准备赋值的数据---时间和当前登录的id(通过ThreadLocal获取id)

        LocalDateTime now = LocalDateTime.now();
        Long currentId = BaseContext.getCurrentId();//当前登录id

        //4.根据当前不同的操作类型,为对应的属性通过反射来赋值
        if(operationType == OperationType.INSERT){
            //为4个公共字段赋值
            try {
                Method setCreateTime = entity.getClass().getDeclaredMethod("setCreateTime", LocalDateTime.class);
                Method setCreateUser = entity.getClass().getDeclaredMethod("setCreateUser", Long.class);
                Method setUpdateTime = entity.getClass().getDeclaredMethod("setUpdateTime", LocalDateTime.class);
                Method setUpdateUser = entity.getClass().getDeclaredMethod("setUpdateUser", Long.class);

                //通过反射为对象属性赋值
                setCreateTime.invoke(entity,now);
                setCreateUser.invoke(entity,currentId);
                setUpdateTime.invoke(entity,now);
                setUpdateUser.invoke(entity,currentId);
            } catch (Exception e) {
                e.printStackTrace();
            }

        }else if(operationType == OperationType.UPDATE){
            //为2个公共字段赋值
            try {
                Method setUpdateTime = entity.getClass().getDeclaredMethod("setUpdateTime", LocalDateTime.class);
                Method setUpdateUser = entity.getClass().getDeclaredMethod("setUpdateUser", Long.class);

                //通过反射为对象属性赋值

                setUpdateTime.invoke(entity,now);
                setUpdateUser.invoke(entity,currentId);
            } catch (Exception e) {
                e.printStackTrace();
            }
        }

    }

}

自定义注解 AutoFill,用于标识需要进行公共字段自动填充的方法

自定义切面类 AutoFillAspect,统一拦截加入了 AutoFill 注解的方法,通过反射为公共字段赋值

在 Mapper 的方法上加入 AutoFill 注解

相关推荐
LSL666_4 小时前
1 概述及简单登录(不涉及数据库)
数据库·servlet
q***06477 小时前
MySQL的UPDATE(更新数据)详解
数据库·mysql
8***B7 小时前
MySQL性能
数据库·mysql
q***72197 小时前
oracle使用PLSQL导出表数据
数据库·oracle
数据库生产实战7 小时前
Oracle DG备库日志切换解析,Private strand flush not complete如何理解?(基础知识)
数据库·oracle
百***75747 小时前
从 SQL 语句到数据库操作
数据库·sql·oracle
i***39587 小时前
SQL 注入详解:原理、危害与防范措施
数据库·sql·oracle
q***69778 小时前
Spring Boot与MyBatis
spring boot·后端·mybatis
m***56728 小时前
Win10下安装 Redis
数据库·redis·缓存
Warren988 小时前
Python自动化测试全栈面试
服务器·网络·数据库·mysql·ubuntu·面试·职场和发展