公共字段自动填充技术(自定义注解)

需求:多次复用功能、解决代码冗余问题。

一、需求分析

实现思路

  • 了解触发自动填充的时机

  • 切面拦截统一赋值

  • 定义注解(方便按需分配)

实现步骤:

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

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

3). 在 Mapper 的方法上加入 AutoFill 注解

技术点:

  • 注解(自定义注解)

  • 枚举(标记操作类型)

  • AOP(自定义切面)

  • 反射(通过反射给公共字段赋值)

二、代码实现

为了更加规范:准备常量类封装方法名

公共字段自动填充技术(自定义注解)_第1张图片

注解常量AutoFillConstant类

/**
 * 公共字段自动填充相关常量
 */
public class AutoFillConstant {
    /**
     * 实体类中的方法名称
     */
    public static final String SET_CREATE_TIME = "setCreateTime";
    public static final String SET_UPDATE_TIME = "setUpdateTime";
    public static final String SET_CREATE_USER = "setCreateUser";
    public static final String SET_UPDATE_USER = "setUpdateUser";
}

1、创建枚举包统一管理枚举

定义哪类操作使用

公共字段自动填充技术(自定义注解)_第2张图片

/**
 * 数据库操作类型
 */
public enum OperationType {
​
    /**
     * 更新操作
     */
    UPDATE,
​
    /**
     * 插入操作
     */
    INSERT
​
}

2、创建注解包统一管理注解

AutoFill定义

/**
 * 自定义注解,用于标识某个方法需要进行功能字段自动填充处理
 */
@Target(ElementType.METHOD)     //指定注解添加的位置(方法上)
@Retention(RetentionPolicy.RUNTIME)  //固定写法
public @interface AutoFill {
    //数据库操作类型:UPDATE INSERT
    OperationType value();
}

3、创建切面包和切面类

  • 实现:统一拦截加入定义注解的方法、通过反射给公共字段赋值。

  • 切面 = 切入点 + 通知

/**
 * 自定义切面,实现公共字段自动填充处理逻辑
 */
@Aspect //切面注解
@Component //实现bean,交给spring容器管理
@Slf4j //方便记录日志
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];
​
        //准备赋值的数据
        LocalDateTime now = LocalDateTime.now();
        Long currentId = BaseContext.getCurrentId();
​
        //根据当前不同的操作类型,为对应的属性通过反射来赋值
        if(operationType == OperationType.INSERT){
            //3、为4个公共字段赋值
            try {
                //获取方法名和参数
                Method setCreateTime = entity.getClass().getDeclaredMethod(AutoFillConstant.SET_CREATE_TIME, LocalDateTime.class);
                Method setCreateUser = entity.getClass().getDeclaredMethod(AutoFillConstant.SET_CREATE_USER, Long.class);
                Method setUpdateTime = entity.getClass().getDeclaredMethod(AutoFillConstant.SET_UPDATE_TIME, LocalDateTime.class);
                Method setUpdateUser = entity.getClass().getDeclaredMethod(AutoFillConstant.SET_UPDATE_USER, 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(AutoFillConstant.SET_UPDATE_TIME, LocalDateTime.class);
                Method setUpdateUser = entity.getClass().getDeclaredMethod(AutoFillConstant.SET_UPDATE_USER, Long.class);
​
                //通过反射为对象属性赋值
                setUpdateTime.invoke(entity,now);
                setUpdateUser.invoke(entity,currentId);
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }
​
}

4、方法上使用注解即可

你可能感兴趣的:(后端)