SpringBoot利用切面注解及反射实现事件监听功能

作者:mabo_9704@163.com 时间:2022-09-25 16:55:00 

前言

当某个事件需要被监听的时候,我们需要去做其他的事前,最简单的方式就是将自己的业务 方法追加到该事件之后。

但是当有N多个这样的需求的时候我们都这样一个个去添加修改事件的源码吗?

这篇文章将告诉你如何用一个注解,就可以将你的业务代码通过切面的方式添加到事件的前后,而不需要修改事件的代码

效果图

如下图所示,add方法内并没有调用其他的方法,但是其他方法仍然被执行了。

只要给监听方法加@AddEventListener()注解就可以让它在事件前后执行了

SpringBoot利用切面注解及反射实现事件监听功能

SpringBoot利用切面注解及反射实现事件监听功能

监听原理

该方法是利用切面、注解、反射来实现SpringBoot的事件监听的

1.通过Aspect的切面,切入事件方法

首先使用Aspec的Around(也可以用before或者after,但是比较麻烦)注解,切入AddEvent的方法中,around注解的方法中,可以在事件方法的执行前后添加业务代码。但是我们不直接加入需要添加的业务,进入第二步骤。

2.利用反射获取被AddEventAop注解的类和方法

利用反射Class.forName(class),获取被AddEventAop注解的类(当然你也可以修改一下,获取所有的类),该类哪个方法被AddEventListener注解了,就执行该方法,则监听执行成功。

method.invoke(o, args);

注意(非常重要)

  • AddEventListener使用的类上,必须被AddEventAop注解了,否则反射的时候方法不会被执行。

  • 事件的类必须是bean,否则切面失败。

  • 监听方法和(被监听方法)事件方法的参数数量,类型,顺序必须一致,否则可能导致反射执行方法失败

核心源码

@Around("@annotation(event)")
   public Object addEventListener(ProceedingJoinPoint joinPoint, AddEventAop event) throws Throwable {
       Object[] args = joinPoint.getArgs();
       //存储需要在方法执行之后再执行的类
       List<Method> afterEventMethod = new ArrayList<>();
       //反射获取AddEventListener修饰的方法并执行
       //获取自定义注解的配置
       final Map<String, Object> beans = applicationContext.getBeansWithAnnotation(AddEventAop.class);
       for (String key : beans.keySet()) {
           //Spring 代理类导致Method无法获取,这里使用AopUtils.getTargetClass()方法
           Object o  = beans.get(key);
           Class<?> aClass = beans.get(key).getClass();
           String name = aClass.getName();
           //aop切面会导致方法注解丢失,在这里处理获取原类名
           if (name.contains("$$")){
               String[] names = name.split("\\$\\$");
               name=names[0];
               aClass = Class.forName(name);
           }
           Method[] methods = aClass.getMethods();
           for (Method method : methods) {
               //获取指定方法上的注解的属性
               AddEventListener annotation = method.getAnnotation(AddEventListener.class);
               if (annotation!=null){
                   //执行所有的注解了该类的方法
                   EventType value = annotation.value();
                   if (value.equals(EventType.BEFOREEVENT)){
                       method.invoke(o, args);
                   }else{
                      afterEventMethod.add(method);
                   }
               }
           }
       }
       //执行被切面的方法
       Object proceed = joinPoint.proceed(args);
       //执行需要在方法执行之后再执行的方法
       for (Method method : afterEventMethod) {
           Class<?> aClass = method.getDeclaringClass();
           Object o = aClass.newInstance();
           method.invoke(o, args);
       }
       return proceed;
   }

源码地址

Github项目地址

来源:https://blog.csdn.net/weixin_47053123/article/details/125812796

标签:SpringBoot,事件监听,切面,注解,反射
0
投稿

猜你喜欢

  • Android中资源文件用法简单示例

    2023-09-24 22:29:04
  • Android Notification 使用方法详解

    2021-07-14 14:42:01
  • java多线程之停止线程的方法实例代码详解

    2023-03-23 04:35:21
  • Java静态代理与动态代理案例详解

    2021-12-09 19:55:44
  • 关于mybatis使用${}时sql注入的问题

    2023-04-18 03:29:40
  • spring boot 使用profile来分区配置的操作

    2022-11-27 22:55:15
  • c# 快速排序算法

    2021-10-18 07:33:20
  • Android WaveView实现水流波动效果

    2021-11-09 16:50:59
  • C#基于NPOI生成具有精确列宽行高的Excel文件的方法

    2022-12-18 13:32:46
  • Java对象在JVM中的生命周期详解

    2023-11-24 16:15:03
  • 深入解析Spring Boot 的SPI机制详情

    2021-07-09 04:26:49
  • SpringCloud使用Feign实现服务调用

    2021-11-10 05:19:20
  • Java动态显示文件上传进度实现代码

    2022-09-09 20:18:17
  • Java之Algorithm_analysis案例详解

    2022-03-07 01:34:50
  • 实例讲解Android中的View类以及自定义View控件的方法

    2023-08-11 06:26:42
  • eclipse创建java项目并运行的详细教程讲解

    2022-09-27 20:15:13
  • 关于C++一些特性的探究

    2022-04-10 07:17:54
  • Spring中的使用@Async异步调用方法

    2023-07-10 17:20:28
  • java实现冒泡排序算法

    2023-10-17 20:44:01
  • 深入学习java8 中的CompletableFuture

    2022-05-19 04:44:26
  • asp之家 软件编程 m.aspxhome.com