菜品管理——公共字段自动填充

问题分析

业务表中存在一些公共字段,导致存在冗余代码,后期维护不便

实现思路

技术点:枚举  注解 AOP  反射

  • 自定义注解 AutoFill,用于标识需要进行公共字段自动填充的方法
  • 自定义切面类AutoFillAspect ,统一拦截加入了AutoFill注解的方法,通过反射为公共字段赋值
  • 在Mapper的方法上加入AutoFill注解

代码开发

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

在common下添加一个包

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

在common下添加一个包

自定义切面类AutoFillAspect

在Mapper的方法上加入AutoFill注解

小测试

咱们通过前后端联调的方式进行测试

首先在前端进行修改

 记得提前打断点,我们会发现程序执行到log处

此时发现拦截到的就是update方法

编写前置通知

ctrl +alt +B 点击Signature

 转换为MethodSignature(其子接口)

getDeclaredMethod 抛出异常

invoke 抛出异常 

修改以后

常量类

代码直接用这些可能会容易出错,而且不规范

所以我们直接使用common中的常量类

 切面类代码

@Before("autoFillPiontCut()")
    public void autoFill(JoinPoint joinPoint){
        log.info("开始进行公共字段的自动填充....");
        //获取当前被拦截到的方法上的数据库操作类型
        MethodSignature signature = (MethodSignature)joinPoint.getSignature();//方法签名对象
        AutoFill autoFill = signature.getMethod().getAnnotation(AutoFill.class);//获得方法上的注解对象
        OperationType operationType = autoFill.value();//获得数据库的操作类型

        //获取被拦截的方法的参数————实体对象
        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){
            //为4个公共字段赋值
            try {
                Method setCreateTime = entity.getClass().getDeclaredMethod(AutoFillConstant.SET_CREATE_TIME, LocalDateTime.class);
                Method setCreteUser = 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);
                setCreteUser.invoke(entity,currentId);
                setUpdateTime.invoke(entity,now);
                setUpdateUser.invoke(entity,currentId);
            } catch (Exception e) {
                throw new RuntimeException(e);
            }

        } 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) {
                throw new RuntimeException(e);
            }
        }


    }

在mapper加入相应的注解

在切面已经为公共属性赋值,那么其他地方就不需要赋值

功能测试

后端运行,进行前后端联调

发现确实拦截到了update方法

没毛病

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值