搞清楚Spring事件机制后:Spring的源码看起来简单多了
本文主讲Spring的事件机制,意图说清楚:
- 什么是观察者模式?
- 自己实现事件驱动编程,对标Spring的事件机制
- 彻底搞懂Spring中的事件机制,从而让大家
本文内容较长,代码干货较多,建议收藏后持续阅读。
Spring框架已然是Javaeee开发领域的霸主,无论是使用SpringBoot还是SpringCloud,都离不开Spring框架。
作为Java开发者,无论是面试求职还是日常开发,就必须得熟练掌握、运用Spring框架。
因此学习Spring框架源码也就成为了大家最重要的事情之一。
Spring框架中用到的设计模式
Spring框架中运用了大量的设计模式,如果对设计模式掌握的不够熟练,阅读源码的时候就会感到很吃力,不明白作者为什么这样写,有时候一个方法一个方法的跟进去,到达一定的深度之后就断了,无法连贯起来,最后不得不放弃。
简单工厂,又叫静态工厂方法模式:Spring中的BeanFactory。
工厂方法模式:Spring中的FactoryBean。
单例模式:Spring中的BeanFactory就是全局单例的,我们日常开发的service默认也是单例模式,单例存在于Spring容器中。
适配器模式:Spring中的AOP、拦截器。
包装器模式:Spring中的各种Wrapper、Decorator。
代理模式:Spring中的AOP就是采用的代理模式,代理有JDK动态代理(JdkDynamicAopProxy)和Cglib代理(Cglib2AopProxy)两种方式。
观察者模式:Spring中的ApplicationListener,事件驱动编程思想。
策略模式:Spring中在实例化对象的时候用到Strategy模式。
模板方法:Spring中的JdbcTemplate、RestTemplate、RedisTemplate等。
熟练掌握以上设计模式可以在阅读Spring、SpringBoot等框架源码的时候,可以做到畅通无阻。
观察者模式
当一个对象发生改变时候,自动通知其他对象作出相关动作。
当一个事件发生时候,自动触发依赖这个事件的其他事件。
这就是观察者模式要做的事情。
举例:当一个开通会员的订单支付完成之后(第三方支付回调过来了),需要更改订单的支付状态、给用户开通会员、如果有分销的逻辑还要触发分润逻辑、给用户开通其他会员附属权益。伪代码如下:
这种做法是最简单也是最low的方法,如果再有其他业务需要依赖订单回调的话,那么需要订单业务开发人员持续的在这里添加相关业务,或同步或异步调用。
那么如果我发布一个订单事件呢?需要监听该事件的其他业务组自己订阅这个事件就好了,订单业务的开发同学再也不用改动自己业务的代码了。
单体应用下我们通常使用事件驱动的编程模式。分布式应用下大家都会采用消息队列发布-订阅的模式去做这块事情。无论采用哪种方式,其核心思想都是观察者模式。
观察者模式中有几个角色:
- 主题:主题中包含多个观察者,以及观察者的添加、删除,同时需要提供触发观察者事件的方法。
- 观察者:也叫监听器,会有多个观察者,不同的观察者监听到事件后做不同的逻辑处理。
自己实现事件驱动编程
先定义好事件的设计,类图如下:
事件定义类图
Event:事件的接口类,可以设置或获取数据EventObject
Event
AbstractEvent:事件抽象实现类,持有EventObject对象,并提供默认实现
AbstractEvent
OrderCallbackEvent:订单回调事件,用于定义具体的事件
OrderCallbackEvent
然后我们来定义事件的监听器(观察者),以及事件广播器(主题),类结构设计如下图:
EventListener:监听器接口类,定义监听器的方法
EventListener
AbstractEventListener:监听器抽象实现类,多了order属性,用于监听器执行顺序
AbstractEventListener
Test1EventListener:事件监听器1
Test1EventListener
Test2EventListener:事件监听器2
Test2EventListener
Test3EventListener:事件监听器3
Test3EventListener
OrderEventMulticaster:事件广播器
EventMulticaster
我们来写一个测试方法,看看当订单完成回调的时候是如何触发这3个监听器的:
运行输出:
可以看见,监听器按照我们给定的order顺序依次执行三个监听器的事件。
上面就是我们自己实现的基于事件驱动的代码,在这个基础上我们还可以针对EventMulticaster广播器再制定特制的OrderEventMulticater以及其他业务的事件广播器。
另外对于每一个监听器我们也可以单独开启一个独立的线程异步去执行监听器里的业务,而不阻塞订单回调事件的执行。
当有新的业务需要订单回调进行触发的时候,就可以再增加一个Listener到监听器集合里面就可以了。不需要再更改订单回调的业务,达到解耦的效果。
Spring的事件机制
Spring中的事件机制有四个关键的类:
ApplicationEvent:事件抽象类,所有的具体事件类都得继承这个类,支持将数据设置到EventObject中:
Spring中的事件有如下几种:
ApplicationListener:事件监听器接口,定义通用方法onApplicationEvent:
ApplicationEventMulticaster:事件广播器接口,用于事件监听器的注册和事件的广播。
ApplicationEventPublisher:事件发布者,调用ApplicationEventMulticaster中的multicastEvent方法触发广播器持有的监听器集合执行onApplicationEvent方法,从而完成事件发布。
Spring中的每一个具体的事件都跟随有一堆的监听器,以及事件的的广播器和发布者。
搞清楚Spring的事件机制可以让我们的Spring的源码阅读之路畅通无阻。
搞清楚Spring事件机制后:Spring的源码看起来简单多了的更多相关文章
- 事件机制-Spring 源码系列(4)
事件机制-Spring 源码系列(4) 目录: Ioc容器beanDefinition-Spring 源码(1) Ioc容器依赖注入-Spring 源码(2) Ioc容器BeanPostProcess ...
- Spring事件机制详解
一.前言 说来惭愧,对应Spring事件机制之前只知道实现 ApplicationListener 接口,就可以基于Spring自带的事件做一些事情(如ContextRefreshedEvent),但 ...
- 深入理解Spring事件机制(一):广播器与监听器的初始化
前言 Spring 从 3.x 开始支持事件机制.在 Spring 的事件机制中,我们可以令一个事件类继承 ApplicationEvent 类,然后将实现了 ApplicationListener ...
- Spring第四天,BeanPostProcessor源码分析,彻底搞懂IOC注入及注解优先级问题!
- 从Chrome源码看浏览器的事件机制
.aligncenter { clear: both; display: block; margin-left: auto; margin-right: auto } .crayon-line spa ...
- 涨姿势:Spring Boot 2.x 启动全过程源码分析
目录 SpringApplication 实例 run 方法运行过程 总结 上篇<Spring Boot 2.x 启动全过程源码分析(一)入口类剖析>我们分析了 Spring Boot 入 ...
- Spring Boot 2.x 启动全过程源码分析
Spring Boot 2.x 启动全过程源码分析 SpringApplication 实例 run 方法运行过程 上面分析了 SpringApplication 实例对象构造方法初始化过程,下面继续 ...
- Spring Boot Dubbo 应用启停源码分析
作者:张乎兴 来源:Dubbo官方博客 背景介绍 Dubbo Spring Boot 工程致力于简化 Dubbo | grep tid | grep -v "daemon" tid ...
- Spring的IOC常用注解(含源码)
一.容器中注入组件 1,包扫描 + 组件标注注解 源码:Demo01_ComponentScan a)组件标注 @Controller @Service @Repository @Component ...
随机推荐
- 异常检测算法Robust Random Cut Forest(RRCF)关键定理引理证明
摘要:RRCF是亚马逊发表的一篇异常检测算法,是对周志华孤立森林的改进.但是相比孤立森林,具有更为扎实的理论基础.文章的理论论证相对较为晦涩,且没给出详细的证明过程.本文不对该算法进行详尽的描述,仅对 ...
- 在微信框架模块中,基于Vue&Element前端,通过动态构建投票选项,实现单选、复选的投票操作
最近把微信框架的前端改造一下,在原来基于Bootstrap框架基础上的微信后台管理,增加一套Vue&Element的前端,毕竟Vue的双向绑定开发起来也还是很方便的,而且Element本身也提 ...
- 03.28,周六,12:00-17:00,ICPC训练联盟周赛,选用试题:UCF Local Programming Contest 2016正式赛。
A. Majestic 10 题意:三个数均大于10则输出"triple-double",如果两个数大于10则输出"double-double",如果一个大于1 ...
- SE_Work4_软件案例分析
项目 内容 课程:北航-2020-春-软件工程 博客园班级博客 要求:分析软件案例 个人博客作业-软件案例分析 班级 005 这个作业在哪个具体方面帮助我实现目标 分析对比一类软件,学会规划分析软件的 ...
- java面试一日一题:java中一个对象实例的结构是什么样子的
问题:请讲下在java程序运行时一个对象实例的数据结构是什么样子的 分析:该问题主要考察对java中对象的理解,在程序运行过程中一个对象实例是以什么样的形式存在的 回答要点: 主要从以下几点去考虑, ...
- Element ui结合springboot的简单实战
Eelment UI简单实战 前端开发 1 创建项目,导入element ui(略) 2 大致设计出想要的效果,如下 3 创建包 根据设计的大致模样在项目的components中创建对应的包,方便以后 ...
- static 静态文件配置
- vue常见错误
错误集锦 错误一 错误二 原因是写太多的import,修改呈如下方式 错误三 源码如下 原因是没有在return后面添加值 应该为 return false
- shell基础之EOF的用法
一.EOF的用法 EOF是(END Of File)的缩写,表示自定义终止符.既然自定义,那么EOF就不是固定的,可以随意设置别名,在linux按ctrl-d 就代表EOF. EOF一般会配合cat能 ...
- Idea项目上传到gitlab(以新建项目为例)
1.首先,需要你自己登录GitLab,并新建一个项目的链接,如下图所示: 图一: 图二: 图三(idea上传时用到此链接): 2.在idea上新建一个demo项目,创建一个Git仓库: 3.点击创建后 ...