欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

基于事件驱动的组件架构

程序员文章站 2022-07-17 09:14:16
本架构主要目的是改进软件开发中松耦合、增加模块的重用性、提高开发效率。 在常规的模块间方法直接调用式开发中,新增的功能对原有模块代码的稳定性、重用性破坏大,不利于软件的后期维护,且开发效率低。 另外,在传统的软件开发方法中,如果新增的功能的逻辑在其它模块需要重复使用,则只能通过copy代码或方法调用 ......

本架构主要目的是改进软件开发中松耦合、增加模块的重用性、提高开发效率。

在常规的模块间方法直接调用式开发中,新增的功能对原有模块代码的稳定性、重用性破坏大,不利于软件的后期维护,且开发效率低。

另外,在传统的软件开发方法中,如果新增的功能的逻辑在其它模块需要重复使用,则只能通过copy代码或方法调用的方式来重用,还是需要改动原代码。
通过本技术方案,可以将一些功能抽像为组件,这些组件可以被动态的插入的现有模块中,不需要改动原有模块代码,从而增加了重用性,提高了开发效率。

技术解决方案

a. 相关概念抽像及定义

首先在软件的业务模块中抽像出核心业务逻辑(core service)及核心业务数据(core data model),同时抽像出核心业务逻辑相对应的事件接口(event interface)。
在核心业务之外的业务被称之为组件(component),组件是由一组业务插件(plugin)组成的,这些插件是一个或多个核心业务的事件的实现体,核心业务和事件的关联,

是通过插件桩(plugin bundle)收集起来的,具体的关联关系是在一个名为component.xml(组件描述文件)中定义的。

 

b. 组件的开发


当需要在核心业务逻辑中插入新的逻辑时,需要定义出相应的插件,使插件实现相应的业务事件接口,在插件中实现具体业务逻辑代码,其需要的核心业务数据会通过事件接口做为参数传递过来,这样插件就可以无入侵式的操作核心业务数据了。

 

 

c. 组件的加载和启动


首先组件有以下状态:安装状态和启动状态,分别为已安装和未安装,已启动和未启动。
当系统启动时,组件加载模块查找所有的component.xml,并解析出插件和核心业务的关联关系,并将这种关系通过组件视图(component view)记录在内存中。
在加载过程中,检测数据中是否有此组件的记录,如果没有则在数据中记录,此时组件的状态被标记为未安装、未启动。如果数据库中已经存在则检测其状态,如果是已启动状态,则将此组件启动,并标记为已启动状态。
在系统的控制台中展示组件的列表,并提供操作按钮,可以执行以下操作:安装/卸载、启动/停用。
综上所述,组件被启动可能通过两种途径:系统加载时或在控制台中手动启动。组件被启动后,根据组件视图中记录的关系将插件和插件桩进行关系绑定,即将插件动态插入到插件桩中,即完成业务逻辑动态地、无侵入式地插入到核心业务逻辑中。

d. 插件调用逻辑说明


在软件运行中,当核心业务逻辑被调用的同时调用事件接口,同时向接口传递核心业务数据,这个过程被称为事件激发。
当事件激时,插件桩里已经收集了相应的插件集合(事件的实现体),进而在插件桩中一个个的调用插件,传递核心数据,这样插件被执行。

 

方案的效果


通过上述组件的技术方案,可以实现对原系统无侵入式的、松耦合式的开发,开发人员无需关系核心代码细节,只要了解事件接口就可以完成对核心业务的扩展,开发效率高、重用性高。
效果举例:以电子商务系统为例,在一个电商业务中创建订单是核心的业务逻辑,在订单创建成功后,可能需要发送一条手机短信给顾客, 基于此种组件方案,开发人员需要开发一款基于某个电信网关的短信发送组件。开发人员不必了解订单创建的具体代码,只需关心如何跟电信网关对接即可,组件机制会将订单信息、顾客手机号等核心数据通过订单的创建事件传递给插件,插件拿到上述数据, 根据具体接入的电信网关参数,将核心数据(订单信息、顾客手机号)组织好文字,传递给电信网关就完成了通过此短信网关发送订单短信的功能。
一旦业务发生变更,比如要更换其它短信网关发送短信,此时再根据具体的网关参数开发另一款短信组件,然后停用之前的短信组件,安装上新的组件即可。
可以看出,从功能开发,到后期的变更维护,订单创建的核心业务代码从未暴露给组件开发人员,他们也不需要去改动核心代码,保持了原有系统的稳定性,降低了新功能开发的难度,而且基于同样规则的系统都可以安装此短信组件,直接使用,极大的降低了开发成本。

基于事件驱动的组件架构

 

 

实际应用

 

 我们以本架构在javashop电商系统中的应用案例说明:

一、订单的创建和短信发送

a. 订单核心业务

 基于事件驱动的组件架构

 

 

 


1.订单核心业务(orderservice)
负责订单的创建,这属于电商业务的核心,通过createorder()方法创建订单。

2.订单插件桩(orderpluginbundle)
在其pluginlist属性中存储了所有响应订单业务事件的插件。

3.订单创建接口
订在订单创建成功后,会激发此事件,而实际调用的是上述pluginlist中的插件。
b. 组件的组成

基于事件驱动的组件架构

 

 

 

 

如上图所示,订单短信组件由以下内容:
1. 订单短信组件(ordersmscomponent)
订单组件类,在组件被安装时会调用其install方法,完成一些安装必要的操作。

2. 订单短信插件
此类实现了订单创建事件,响应了订单完成事件(onordercreate)方法,当订单创建完成时,会调用此方法。

3. 组件的描述文件component.xml
此描述文件定义了订单插件(ordersmsplugin)要注入到订单插件桩中,即定义了插件要对应的核心业务


c. 组件加载、启动过程

基于事件驱动的组件架构

 

 

组件的加载、启动过程细节如下:
当系统启动时,组件加载模块查找所有的,并解析出插件和核心业务的关联关系,并将这种关系通过组件视图记录在内存中。
在加载过程中,检测数据中是否有此组件的记录,如果没有则在数据中记录,此时组件的状态被标记为未安装、未启动。如果数据库中已经存在则检测其状态,如果是已启动状态,则将此组件启动。
组件启动首先将插件注入到相应的插件桩中,此时当事件被激发时,通过插件桩就可以调用到实现了此事件的所有插件了。组织好插件的关系后,将此组件在数据库中标记为已启动。

d. 事件激发及插件调用

基于事件驱动的组件架构

 

 


如上图所示,描述了事件激发和插件调用的过程,具体技术实施细节如下:
在这里循环所有的订单创建事件调用之(激发事件接口),这些事件实际是被注入进来的、具体的插件实现体,比如订单短信插件,此时订单短信插件被动态调用,在此插件中完成发送短信的逻辑。

通过上述架构的实施,可以达到如下效果: 

现在短信的厂商众多,我们分别为其实现“短信发送插件”,这对于“订单核心逻辑”代码是不需要改动的,

一方面提升了核心代码的稳定,一方面增加可对“新短信厂商”的扩展性。

 

后记

通过这种组件架构,其实在电商架构领域中还有很多模块可以采用组件式的架构,以javashop电商系统中的应用为例:

支付组件

订单的支付为核心逻辑,他来处理状态改变,记录日志等。

通过调用支付插件来完成具体的支付操作,传递的是支付金额,订单号核心标准数据。

而支付插件可以多种实现:支付宝、微信、银联等等。

这样增强了订单核心的稳定,提高了支付方式的扩展性

 

物流查询

查询动作和展示是核心,传递给插件“物流公司”,“物流单号”核心标准数据

物流查询有多种插件实现:kuaidi100,showapi,菜鸟等等。

同样增强了订单核心的稳定,提高了物流查询厂商的扩展性

 

图片存储

图片都上传和显示是核心,传递给插件图片流是核心标准数据

图片的存储有多种插件实现:如阿里oss,七牛、又拍等等。

同样增强了系统核心的稳定,提高了图片存储厂商的扩展性

 

其实还有很多逻辑可以采用这种思路,提高稳定性,提高扩展性,在这里就不一一列举

这需要对自己的系统功能进行分析抽象,抽象出核心逻辑、插件接口标准、标准数据。

易族智汇(javashop)原创文章