Spring事件机制源码分析

偏执的太偏执、 2022-02-27 06:18 410阅读 0赞

前言

由于之前使用Spring事件机制出现了问题,所以特意去了解这块的源码。Spring事件机制其实就是事件发布/订阅(注意在Spring中订阅指的是监听)。

PS:Spring版本为5.1.5.RELEASE

源码分析

初始化

初始化这块关键是核心组件的注册

  1. ApplicationEventPublisher的初始化与注册,关键方法为AbstractApplicationContext的方法prepareBeanFactory()
  2. ApplicationEventMulticaster的初始化与注册,关键方法为AbstractApplicationContextinitApplicationEventMulticaster()方法
  3. ApplicationListener的初始化与注册,关键方法为AbstractApplicationContextregisterListeners()方法

这块不细说,感兴趣的可以自行跟踪关键方法

事件发布/订阅

事件发布/订阅的关键方法为AbstractApplicationContextpublishEvent,源码如下:

  1. protected void publishEvent(Object event, ResolvableType eventType) {
  2. // 避免空指针
  3. Assert.notNull(event, "Event must not be null");
  4. if (logger.isTraceEnabled()) {
  5. logger.trace("Publishing event in " + getDisplayName() + ": " + event);
  6. }
  7. // 处理event对象,将其转换为ApplicationEvent
  8. ApplicationEvent applicationEvent;
  9. if (event instanceof ApplicationEvent) {
  10. applicationEvent = (ApplicationEvent) event;
  11. }
  12. else {
  13. applicationEvent = new PayloadApplicationEvent<Object>(this, event);
  14. if (eventType == null) {
  15. eventType = ((PayloadApplicationEvent) applicationEvent).getResolvableType();
  16. }
  17. }
  18. // 是否延迟多播,即将事件发布到所有监听器中
  19. if (this.earlyApplicationEvents != null) {
  20. this.earlyApplicationEvents.add(applicationEvent);
  21. }
  22. else {
  23. //此处为事件监听处理器的调用关键
  24. getApplicationEventMulticaster().multicastEvent(applicationEvent, eventType);
  25. }
  26. // 是否将事件发布到父容器中
  27. if (this.parent != null) {
  28. if (this.parent instanceof AbstractApplicationContext) {
  29. ((AbstractApplicationContext) this.parent).publishEvent(event, eventType);
  30. }
  31. else {
  32. this.parent.publishEvent(event);
  33. }
  34. }
  35. }

通过代码跟踪,发现Spring中使用ApplicationEventMulticaster的默认实现SimpleApplicationEventMulticaster来触发事件的监听,关键方法为multicastEvent()方法,源码如下:

  1. @Override
  2. public void multicastEvent(final ApplicationEvent event, ResolvableType eventType) {
  3. // 获取事件类型
  4. ResolvableType type = (eventType != null ? eventType : resolveDefaultEventType(event));
  5. for (final ApplicationListener<?> listener : getApplicationListeners(event, type)) {//依次遍历事件监听器
  6. // 获取线程池
  7. Executor executor = getTaskExecutor();
  8. if (executor != null) {//线程池不为null,则异步调用监听器
  9. executor.execute(new Runnable() {
  10. @Override
  11. public void run() {
  12. invokeListener(listener, event);
  13. }
  14. });
  15. }
  16. else {// 同步调用监听器
  17. invokeListener(listener, event);
  18. }
  19. }
  20. }

发表评论

表情:
评论列表 (有 0 条评论,410人围观)

还没有评论,来说两句吧...

相关阅读