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注解的方法。

相关推荐
GetcharZp5 小时前
告别 jq 噩梦!这款 JSON 神器 fx 让你在终端体验“丝滑”的数据操作
后端
TeDi TIVE5 小时前
开源模型应用落地-工具使用篇-Spring AI-高阶用法(九)
人工智能·spring·开源
MY_TEUCK5 小时前
Sealos 平台部署实战指南:结合 Cursor 与版本发布流程
java·人工智能·学习·aigc
我爱cope6 小时前
【从0开始学设计模式-10| 装饰模式】
java·开发语言·设计模式
朝新_6 小时前
【Spring AI 】图像与语音模型实战
java·人工智能·spring
小码哥_常6 小时前
告别臃肿!Elasticsearch平替Manticore登场
后端
RH2312117 小时前
2026.4.16Linux 管道
java·linux·服务器
zmsofts7 小时前
java面试必问13:MyBatis 一级缓存、二级缓存:从原理到脏数据,一篇讲透
java·面试·mybatis
苍何7 小时前
万字保姆级教程:Hermes+Kimi K2.6 打造7x24h Agent军团
后端
我叫黑大帅7 小时前
为什么map查找时间复杂度是O(1)?
后端·算法·面试