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

相关推荐
庞轩px8 小时前
第七篇:Spring扩展点——如何优雅地介入Bean的创建流程
java·后端·spring·bean·aware·扩展点
ltl8 小时前
Q/K/V 三件套:把 Bahdanau 抽象成一个公式
后端
tongluowan0079 小时前
一个请求在Spring MVC 中是怎么流转的
java·spring·mvc
千叶风行10 小时前
Text-to-SQL 技术设计与注意事项
前端·人工智能·后端
夜郎king10 小时前
Spring AI 对接大模型开发易错点总结与实战解决办法
java·人工智能·spring
oradh10 小时前
Oracle数据库中的Java概述
java·数据库·oracle·sql基础·oracle数据库java概述
组合缺一10 小时前
Java AI 框架三国杀:Solon AI vs Spring AI vs LangChain4j 深度对比
java·人工智能·spring·ai·langchain·llm·solon
阿kun要赚马内10 小时前
后端数据操作组合:Pydantic与ORM
后端·python·orm·sqlalchemy
c++之路11 小时前
适配器模式(Adapter Pattern)
java·算法·适配器模式
吴声子夜歌11 小时前
Java——接口的细节
java·开发语言·算法