Spring事件监听机制(三)

为了理解@EvenListener注解的底层原理,我们可以自己实现一个类似的注解模拟实现。

1.定义@MyListener注解

java 复制代码
    @Target({ElementType.METHOD})
    @Retention(RetentionPolicy.RUNTIME)
    public @interface MyListener {

    }

2.注解使用

java 复制代码
    @Component
    static class SmsService {
        private static final Logger log = LoggerFactory.getLogger(SmsService.class);

        @MyListener
        public void listener(MyEvent event) {
            log.debug("发送短信");
        }
    }

3.注解解析

java 复制代码
public static void main(String[] args) {
        AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(TestMyListener.class);
        SmsService smsService = context.getBean(SmsService.class);
        for(Method method : SmsService.class.getMethods()){
            if(method.isAnnotationPresent(MyListener.class)){
                ApplicationListener listener = new ApplicationListener<MyEvent>() {
                    @Override
                    public void onApplicationEvent(MyEvent event) {
                        try {
                            method.invoke(smsService, event);
                        }catch (Exception e){
                            e.printStackTrace();
                        }
                    }
                };
                context.addApplicationListener(listener);
            }
        }
        context.getBean(MyService.class).doBusiness();
        context.close();
    }

1)获取监听器类

2)通过反射拿到方法

判断方法上的注解是否是我们自定义的注解,如果是,创建ApplicationListener对象(这里使用了泛型去指定事件类型,如果不这样做可能会接收到别的事件而报错,比如说容器关闭事件),重写里面的监听事件的方法,通过反射调用加了@MyListener注解的方法。

3)把监听器加入容器里面。

打印结果:

@MyListener注解实现了事件监听机制。

4.改进点

上面我们固定解析了监听类为SmsService类,现实情况是其它类上也可能加了@MyListener注解,我们可以做得更通用一些。

(1)再写一个监听类EmailService。

java 复制代码
    @Component
    static class EmailService {
        private static final Logger log = LoggerFactory.getLogger(EmailService.class);

        @MyListener
        public void listener(MyEvent event) {

            log.debug("发送邮件");
        }
    }

2)遍历所有满足条件的bean

java 复制代码
public static void main(String[] args) {
        AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(TestMyListener.class);
        for (String beanName : context.getBeanDefinitionNames()){
            Object bean = context.getBean(beanName);
            for(Method method : bean.getClass().getMethods()){
                if(method.isAnnotationPresent(MyListener.class)){
                    ApplicationListener listener = new ApplicationListener<MyEvent>() {
                        @Override
                        public void onApplicationEvent(MyEvent event) {
                            try {
                                method.invoke(bean, event);
                            }catch (Exception e){
                                e.printStackTrace();
                            }
                        }
                    };
                    context.addApplicationListener(listener);
                }
            }
        }
        context.getBean(MyService.class).doBusiness();
        context.close();
    }

3)测试结果

容器监听了所有实现@MyListener注解的方法。

相关推荐
沉木渡香36 分钟前
VSCode中Java开发环境配置的三个层级(Windows版)1-3
java·windows·vscode
程序员小白条38 分钟前
度小满运维开发一面
java·运维·python·职场和发展·运维开发
Leo655356 小时前
JDK8 的排序、分组求和,转换为Map
java·开发语言
书源丶8 小时前
二十八、API之《System 类》——与系统交互的“桥梁”
java·交互
Pluchon8 小时前
硅基计划4.0 算法 字符串
java·数据结构·学习·算法
野生技术架构师8 小时前
1000 道 Java 架构师岗面试题
java·开发语言
青柠编程9 小时前
基于Spring Boot的选课管理系统架构设计
java·spring boot·后端
Mr.wangh9 小时前
Redis主从复制
java·数据库·redis
Porunarufu9 小时前
JAVA·顺序逻辑控制
java·开发语言
1710orange9 小时前
java设计模式:适配器模式
java·设计模式·适配器模式