ApplicationEvent&ApplicationListener

1、参考

Standard and Custom Events

2、前言

公司项目中有个发送站内信的功能,是使用ApplicationEventApplicationListener实现的,对于这一块,之前未接触过,在该项目中,其具体功能如下:

  • ApplicationContext发送ApplicationEvent类型的站内信
  • ApplicationListener接收到站内信信息,进行入库处理,createTime直接写死成了new Date()
  • 用户界面读取入库后的数据,根据createTime倒序呈现给用户

产品经理要求站内信需要按发送的顺序呈现给用户,那问题来了:

ApplicationListener接收到Event信息时,是否是有序的呢?

3、Demo

基于spring boot创建一个简易的ApplicationEvent使用demo,基于该demo执行并发测试,初步验证是否为有序接收

3.1、CustomEvent

自定义Event,继承ApplicationEvent,并实现构造方法

public class CustomEvent extends ApplicationEvent {
    /**
     * Create a new ApplicationEvent.
     *
     * @param source the object on which the event initially occurred (never {@code null})
     */
    public CustomEvent(Object source) {
        super(source);
    }
}

3.2、CustomPublisher

Event事件发布者,此处定义publish方法作为发送入口,提供给controller调用,方便测试

@Component
public class CustomPublisher {

    @Autowired
    private ApplicationContext applicationContext;

    public void publish(CustomEvent customEvent) {
        applicationContext.publishEvent(customEvent);
    }
}

除了使用注解的方式,spring也提供了实现ApplicationEventPublisherAware接口的方式,具体可戳参考链接

3.3、CustomListener

Event事件监听器,接收Event事件并对其进行处理,此处进行简单的打印

@Component
public class CustomListener {
    @EventListener
    public void listen(CustomEvent customEvent) {
        System.out.println(customEvent.getSource());
    }
}

除了使用注解的方式,spring也提供了ApplicationListener接口的方式,具体可戳参考链接

3.4、测试

@RestController
public class RestController {
    @Autowired
    private CustomPublisher customPublisher;
    int i = 0;

    @GetMapping(value = "/publish")
    public void publish() {
        CustomEvent customEvent = new CustomEvent(i++);
        customPublisher.publish(customEvent);
    }
}

使用postmanrunner功能,创建如下runner,进行并发测试:

runner

结果

从结果来看,事件接收是有序的,查阅官网,可以看到这么一句话:

You can register as many event listeners as you wish, but note that, by default, event listeners receive events synchronously. This means that the publishEvent() method blocks until all listeners have finished processing the event.
大致意思是:event listeners默认上同步接收events,这意味着,publicEvent()方法将会阻塞直至所有的listeners处理完event为止。

4、源码

追溯源码,查看ApplicationEvent的发布和接收功能是如何实现的,首先,程序入口位于ApplicationContext#publishEvent

    default void publishEvent(ApplicationEvent event) {
        publishEvent((Object) event);
    }

方法内调用了重载的publicEvent,而该重载publishEvent是一个接口方法,有以下实现:

publishEvent实现

显然,具体的实现类是AbstractApplicationContext#publishEvent

    @Override
    public void publishEvent(Object event) {
        publishEvent(event, null);
    }

继续调用重载方法publishEvent(Object event, @Nullable ResolvableType eventType)

protected void publishEvent(Object event, @Nullable ResolvableType eventType) {
        Assert.notNull(event, "Event must not be null");
        if (logger.isTraceEnabled()) {
            logger.trace("Publishing event in " + getDisplayName() + ": " + event);
        }

        // Decorate event as an ApplicationEvent if necessary
        ApplicationEvent applicationEvent;
        if (event instanceof ApplicationEvent) {
            applicationEvent = (ApplicationEvent) event;
        }
        else {
            applicationEvent = new PayloadApplicationEvent<>(this, event);
            if (eventType == null) {
                eventType = ((PayloadApplicationEvent) applicationEvent).getResolvableType();
            }
        }

        // Multicast right now if possible - or lazily once the multicaster is initialized
        if (this.earlyApplicationEvents != null) {
            this.earlyApplicationEvents.add(applicationEvent);
        }
        else {
            getApplicationEventMulticaster().multicastEvent(applicationEvent, eventType);
        }

        // Publish event via parent context as well...
        if (this.parent != null) {
            if (this.parent instanceof AbstractApplicationContext) {
                ((AbstractApplicationContext) this.parent).publishEvent(event, eventType);
            }
            else {
                this.parent.publishEvent(event);
            }
        }
    }

逻辑可以总结为以下三点:

  • 将入参event包装成applicationEvent
  • 交由多播器进行消息发布(multicastEvent
  • 若父级上下文不为空,则父级上下文同样需要进行消息发布

显然,核心方法在于multicastEvent

    @Override
    public void multicastEvent(final ApplicationEvent event, @Nullable ResolvableType eventType) {
        ResolvableType type = (eventType != null ? eventType : resolveDefaultEventType(event));
        for (final ApplicationListener<?> listener : getApplicationListeners(event, type)) {
            Executor executor = getTaskExecutor();
            if (executor != null) {
                executor.execute(() -> invokeListener(listener, event));
            }
            else {
                invokeListener(listener, event);
            }
        }
    }

这里使用了线程池,调用了invokeListener

    protected void invokeListener(ApplicationListener<?> listener, ApplicationEvent event) {
        ErrorHandler errorHandler = getErrorHandler();
        if (errorHandler != null) {
            try {
                doInvokeListener(listener, event);
            }
            catch (Throwable err) {
                errorHandler.handleError(err);
            }
        }
        else {
            doInvokeListener(listener, event);
        }
    }

继续调用doInvokeListener

private void doInvokeListener(ApplicationListener listener, ApplicationEvent event) {
        try {
            listener.onApplicationEvent(event);
        }
        catch (ClassCastException ex) {
            // ……
        }
    }

onApplicationEvent,是ApplicationListener接口类中的方法,在3.3节中,有提及自定义监听器可以继承ApplicationListener接口并实现onApplicationEvent方法进行监听,若是使用该种方法,则事件的发送和接收处理的源码就已经追溯完毕,不过本文中使用的是注解,故需要继续跟踪下去,onApplicationEvent的实现类颇多:

onApplicationEvent实现类

通过debug验证,ApplicationListenerMethodAdapter才是注解类listener的具体实现:

    public void onApplicationEvent(ApplicationEvent event) {
        processEvent(event);
    }

查看processEvent

    public void processEvent(ApplicationEvent event) {
        Object[] args = resolveArguments(event);
        if (shouldHandle(event, args)) {
            Object result = doInvoke(args);
            if (result != null) {
                handleResult(result);
            }
            else {
                logger.trace("No result object given - no result to handle");
            }
        }
    }

可以看到核心在于doInvoke

    protected Object doInvoke(Object... args) {
        Object bean = getTargetBean();
        ReflectionUtils.makeAccessible(this.method);
        try {
            return this.method.invoke(bean, args);
        }
                // ……
      }

这里使用了反射,调用method方法,那么,method是什么时候赋值,具体值又是什么呢,对method进行Find Usages,可以看到:

    public ApplicationListenerMethodAdapter(String beanName, Class<?> targetClass, Method method) {
        this.beanName = beanName;
        this.method = BridgeMethodResolver.findBridgedMethod(method);
        // ……
    }

method是在ApplicationListenerMethodAdapter构造方法中赋值的,debug,可以看到具体值为:

public void com.kungyu.rabbitmq.CustomListener.listen(com.kungyu.rabbitmq.CustomEvent)

也就是我们自定义的监听方法,那么,ApplicationListenerMethodAdapter构造方法何时被调用,继续Find Usages

public class DefaultEventListenerFactory implements EventListenerFactory, Ordered {

    // ……
    @Override
    public ApplicationListener<?> createApplicationListener(String beanName, Class<?> type, Method method) {
        return new ApplicationListenerMethodAdapter(beanName, type, method);
    }

}

createApplicationListener进行Find Usages

protected void processBean(
            final List<EventListenerFactory> factories, final String beanName, final Class<?> targetType) {
    // ……
    ApplicationListener<?> applicationListener =
                                    factory.createApplicationListener(beanName, targetType, methodToUse);
    // ……
}

processBean进行Find Usages,可以看到是在afterSingletonsInstantiated进行调用的,而afterSingletonsInstantiated是类实例化相关的内容,此处不予展开

5、总结

从测试结果来看,ApplicationListener是有序接收消息的,从源码来看,其实现方式也不算过于复杂,是对知识点的一个很好的补充。另外,在中间件横行的时代,业务角度上貌似这一功能没怎么被使用,出于解耦的目的,应该都会用MQ实现该功能了。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,444评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,421评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,363评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,460评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,502评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,511评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,280评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,736评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,014评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,190评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,848评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,531评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,159评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,411评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,067评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,078评论 2 352

推荐阅读更多精彩内容