Activiti是业界很流行的java工作流引擎,关于Activiti与JBPM5的关系和如何选择不是本文要讨论的话题,相关内容可以baidu一下。Activiti从架构角度看是比较优秀的,是很面向对象的,是我所阅读过的代码结构很棒的开源软件,个人认为比Spring,Hibernate的要好。
Activiti的基础编程框架
![点击查看原始大小图片](http://dl.iteye.com/upload/attachment/0061/8629/bfc57552-9d60-3483-9577-3503688de2d4.jpg)
Activiti基于Spring,ibatis等开源中间件作为软件平台,在此之上构建了非常清晰的开发框架。上图列出了Activiti的核心组件。
1.ProcessEngine:流程引擎的抽象,对于开发者来说,它是我们使用Activiti的facade,通过它可以获得我们需要的一切服务。
2.XXService(TaskService,RuntimeService,RepositoryService...):Activiti按照流程的生命周期(定义,部署,运行)把不同阶段的服务封装在不同的Service中,用户可以非常清晰地使用特定阶段的接口。通过ProcessEngine能够获得这些Service实例。TaskService,RuntimeService,RepositoryService是非常重要的三个Service:
TaskService:流程运行过程中,与每个任务节点相关的接口,比如complete,delete,delegate等等
RepositoryService:流程定义和部署相关的存储服务。
RuntimeService:流程运行时相关服务,如startProcessInstanceByKey.
关于ProcessEngine和XXService的关系,可以看下面这张图:
![](http://dl.iteye.com/upload/attachment/0062/2195/707d4a4c-2c4e-3a1c-83c4-895e22babca0.jpg)
3.CommandContextIntercepter(CommandExecutor):Activiti使用命令模式作为基础开发模式,上面Service中定义的各个方法都对应相应的命令对象(xxCmd), Service把各种请求委托给xxCmd,xxCmd来决定命令的接收者,接收者执行后返回结果。而CommandContextIntercepter顾名思义,它是一个拦截器,拦截所有命令,在命令执行前后执行一些公共性操作。比如CommandContextIntercepter的核心方法:
- public<T>Texecute(Command<T>command){
- CommandContextcontext=commandContextFactory.createCommandContext(command);
- try{
- Context.setCommandContext(context);
- Context.setProcessEngineConfiguration(processEngineConfiguration);
- returnnext.execute(command);
- }catch(Exceptione){
- context.exception(e);
- }finally{
- try{
- context.close();
- }finally{
- Context.removeCommandContext();
- Context.removeProcessEngineConfiguration();
- }
- }
- returnnull;
- }
关于命令模式的细节说明,网上有很多资料,这里不展开。我只是想说一下我看到Activiti的这种设计之后的两点感受:
1)一个产品或者一个项目,从技术上必须有一个明确的、唯一的开发模型或者叫开发样式(真不知道怎么说恰当),我们常说希望一个团队的所有人写出的代码都有统一的风格,都像是一个人写出来的,很理想化,但做到很难,往往我们都是通过“规范”去约束大家这样做,而规范毕竟是程序之外的东西,主观性很强,不遵守规范的情况屡屡发生。而如果架构师给出了明确的开发模型,并使用一些基础组件加以强化,把程序员要走的路规定清楚,那你想不遵守规范都会很难,因为那意味着你写的东西没发工作。就像Activiti做的这样,明确以Command作为基本开发模型,辅之以Event-Listener,这样编程风格的整体性得到了保证。
2)使用命令模式的好处,我这里体会最深的就是 职责分离,解耦。有了Command,各个Service从角色上说只是一些协调者或者控制者,他不需要知道具体怎么做,他只是把任务交给了各个命令。直接的好处是臃肿的、万能的大类没有了。而这往往是我们平时开发中最深恶痛绝的地方。
4.核心业务对象(Task,ProcessInstance,Execution...):org.activiti.engine.impl.persistence.entity包下的类是Activiti的核心业务对象。它们是真正的对象,而不是只有数据没有行为的假对象,搞java企业级开发的人也许已经习惯了下面的层次划分:controller->service->dao->entity, entity只是ORMapping中数据表的java对象体现,没有任何行为(getter/setter不能算行为),对于面向对象来说,这当然是有问题的,记得曾听人说过这样的话“使用面向对象语言进行设计和开发
与 面向对象的设计和开发 是两回事”,面向对象讲究的是“封装”,“多态”,追求的是满足“开-闭”原则的、职责单一的对象社会。如果你认同上述观点,那么相信Activiti会让你感觉舒服一些,以TaskEntity为例,其UML类图如下:
![](http://dl.iteye.com/upload/attachment/0061/8946/a9b586ba-e59a-3375-b271-a3502b04c9ed.jpg)
(图2:TaskEntity类)
TaskEntity实现了3个接口:Task,DelegateTask和PersistentObject。其中PersistentObject是一个声明接口,表明TaskEntity需要持久化。接口是一种角色的声明,是一份职责的描述而TaskEntity就是这个角色的具体扮演者,因此TaskEntity必须承担如complete,delegate等职责。
但是这里有些遗憾的是像complete这么重要的行为居然没有在3个接口中描述(难道是因为工期紧张?^_^),因此“面向抽象”编程对于TaskEntity来说还没有完全做到。但至少Activiti告诉我们:
1)牢记面向抽象编程,把职责拆分为不同的接口,让接口来体现对象的职责,而不用去关心这份职责具体由哪个对象实现;
2)entity其实可以也应该是真正的对象。
5.Activiti的上下文组件(Context)
上下文(Context)用来保存生命周期很长的、全局性的信息。Activiti的Context类(在org.activiti.engine.impl.context包下)保持如下三类信息:
![](http://dl.iteye.com/upload/attachment/0061/8965/02a3161f-5de4-37a4-bfde-877a2a39aab4.jpg)
(图3:Context类)
CommandContext:命令上下文,保持每个命令需要的必要资源,如持久化需要的session。
ProcessEngineConfigurationImpl:流程引擎相关的配置信息。它是整个引擎的全局配置信息,mailServerHost,DataSource等。单例。该实例在流程引擎创建时被实例化,其调用stack如下图:
![](http://dl.iteye.com/upload/attachment/0061/8973/6a55adaf-8247-3174-a48b-4881113f0fed.jpg)
(图4:ProcessEngineConfiguration的初始化)
ExecutionContext:刚看到这个类感觉有些奇怪,不明白其作用是什么。看其代码持有ExecutionEntity这个实例。而ExecutionEntity是Activiti中非常重要的一个类,//TODO
6.Activiti的持久化框架(组件)
Activiti使用ibatis作为ORMapping工具。在此基础之上Activiti设计了自己的持久化框架,看一张图:
![](http://dl.iteye.com/upload/attachment/0061/8986/39c3e1d3-8f00-30e1-96db-f3de6ecce390.jpg)
(图5:Activiti持久化框架)
顶层接口是Session和SessionFactory,这都非常好理解了。
Session有两个实现类:
DbSqlSession:简单点说,DbSqlSession负责sql表达式的执行。
AbstractManager:简单点说,AbstractManager及其子类负责面向对象的持久化操作
同理DbSqlSessionFactory与GenericManagerFactory的区别就很好理解了。
持久化框架也是在流程引擎建立时初始化的,具体见图4.
7.Event-Listener 组件
Activiti允许客户端代码介入流程的执行。为此提供了一个基础组件,看图:
![点击查看原始大小图片](http://dl.iteye.com/upload/attachment/0061/8990/66cecac2-d837-392a-9636-e5b5f532868f.jpg)
(图6:用户代码介入流程的基础组件)
用户可以介入的代码类型包括:TaskListener,JavaDelegate,Expression,ExecutionListener。
ProcessEngineConfigurationImpl持有DelegateInterceptor的某个实例,这样就可以随时非常方便地调用handleInvocation
8.Cache 组件
对Activiti的cache实现很感兴趣,但现在我了解到的情况(也许还没有了解清楚)其cache的实现还是很简单的,在DbSqlSession中有cache实现:
- protectedList<PersistentObject>insertedObjects=newArrayList<PersistentObject>();
- protectedMap<Class<?>,Map<String,CachedObject>>cachedObjects=newHashMap<Class<?>,Map<String,CachedObject>>();
- protectedList<DeleteOperation>deletedObjects=newArrayList<DeleteOperation>();
- protectedList<DeserializedObject>deserializedObjects=newArrayList<DeserializedObject>();
也就是说Activiti就是基于内存的List和Map来做缓存的。具体怎么用的呢?以DbSqlSession.selectOne方法为例:
- publicObjectselectOne(Stringstatement,Objectparameter){
- statement=dbSqlSessionFactory.mapStatement(statement);
- Objectresult=sqlSession.selectOne(statement,parameter);
- if(resultinstanceofPersistentObject){
- PersistentObjectloadedObject=(PersistentObject)result;
- 缓存处理
- result=cacheFilter(loadedObject);
- }
- returnresult;
- }
- protectedPersistentObjectcacheFilter(PersistentObjectpersistentObject){
- PersistentObjectcachedPersistentObject=cacheGet(persistentObject.getClass(),persistentObject.getId());
- if(cachedPersistentObject!=null){
- returncachedPersistentObject;
- }
- cachePut(persistentObject,true);
- returnpersistentObject;
- }
- protectedCachedObjectcachePut(PersistentObjectpersistentObject,booleanstoreState){
- Map<String,CachedObject>classCache=cachedObjects.get(persistentObject.getClass());
- if(classCache==null){
- classCache=newHashMap<String,CachedObject>();
- cachedObjects.put(persistentObject.getClass(),classCache);
- }
- CachedObjectcachedObject=newCachedObject(persistentObject,storeState);
- classCache.put(persistentObject.getId(),cachedObject);
- returncachedObject;
- }
看了Activiti的缓存设计,我现在最大的疑问是Activiti貌似不支持cluster,因为其缓存设计是基于单机内存的,这个问题还需要进一步调查。
9.异步执行组件
Activiti可以异步执行job(具体例子可以看一下ProcessInstance startProcessInstanceByKey(String processDefinitionKey);),下面简单分析一下其实现过程,还是先看图:
![](http://dl.iteye.com/upload/attachment/0061/9029/c5c40965-865e-336b-90ae-354fe3297bda.jpg)
(图7:异步执行组件核心类)
JobExecutor是异步执行组件的核心类,其包含三个主要属性:
1)JobAcquisitionThread jobAccquisitionThread:执行任务的线程 extends java.lang.Thread
2)BlockingQueue<Runnable> threadPoolQueue
3)ThreadPoolExecutor threadPoolExecutor:线程池
方法ProcessEngines在引擎启动时调用JobExecutor.start,JobAcquisitionThread 线程即开始工作,其run方法不断循环执行AcquiredJobs中的job,执行一次后线程等待一定时间直到超时或者JobExecutor.jobWasAdded方法因为有新任务而被调用。
这里发现有一处设计的不够好:JobAcquisitionThread 与JobExecutor之间的关系是如此紧密(你中有我,我中有你),那么可以把JobAcquisitionThread 作为JobExecutor的内部类来实现,同时把ThreadPoolExecutor threadPoolExecutor交给JobAcquisitionThread 来管理,JobExecutor只负责接受任务以及启动、停止等更高级的工作,具体细节委托给JobAcquisitionThread ,责任分解,便于维护,JobExecutor的代码也会看起来更清晰。
10.PVM
PVM:Process Virtal Machine,流程虚拟机API暴露了流程虚拟机的POJO核心,流程虚拟机API描述了一个工作流流程必备的组件,这些组件包括:
PvmProcessDefinition:流程的定义,形象点说就是用户画的那个图。静态含义。
PvmProcessInstance:流程实例,用户发起的某个PvmProcessDefinition的一个实例,动态含义。
PvmActivity:流程中的一个节点
PvmTransition:衔接各个节点之间的路径,形象点说就是图中各个节点之间的连接线。
PvmEvent:流程执行过程中触发的事件
以上这些组件很好地对一个流程进行了建模和抽象。每个组件都有很清晰的角色和职责划分。另外,有了这些API,我们可以通过编程的方式,用代码来“画”一个流程图并让他run起来,例如:
- PvmProcessDefinitionprocessDefinition=newProcessDefinitionBuilder()
- .createActivity("a").initial().<strongstyle="background-color:#ff0000;">behavior</strong>(newWaitState())
- .transition("b").endActivity().createActivity("b")
- .behavior(newWaitState()).transition("c").endActivity()
- .createActivity("c").behavior(newWaitState()).endActivity()
- .buildProcessDefinition();
- PvmProcessInstanceprocessInstance=processDefinition
- .createProcessInstance();
- processInstance.start();
- PvmExecutionactivityInstance=processInstance.findExecution("a");
- assertNotNull(activityInstance);
- activityInstance.signal(null,null);
- activityInstance=processInstance.findExecution("b");
- assertNotNull(activityInstance);
- activityInstance.signal(null,null);
- activityInstance=processInstance.findExecution("c");
- assertNotNull(activityInstance);
以上代码都很简单,很好理解,只有一点需要说明一下,粗体红色背景的behavior方法,为一个PvmActivity增加ActivityBehavior,这是干什么呢?ActivityBehavior是一个interface,其接口声明很简单:
- publicinterfaceActivityBehavior{
- voidexecute(ActivityExecutionexecution)throwsException;
- }
我的理解:Activiti把完成一个PvmActivity的行为单独建模封装在ActivityBehavior中。execute方法只有一个参数ActivityExecution,为啥这么设计?
为了更好地理解ActivityBehavior的作用,我们以TaskEntity.complete方法为例,分析其执行过程,先看complete的代码:
- publicvoidcomplete(){
- fireEvent(TaskListener.EVENTNAME_COMPLETE);
- Context
- .getCommandContext()
- .getTaskManager()
- .deleteTask(this,TaskEntity.DELETE_REASON_COMPLETED,false);
- if(executionId!=null){
- getExecution().signal(null,null);
- }
- }
代码很简单,也很好理解(可能出乎我们的意料,因为完成一个task,其实有很多事情要做的):
1.fireEvent:通知Listener,本任务完成了。
2.数据持久化相关的动作
3.getExecution().signal(null, null):发信号,这里面隐藏的东西就多了,总体来说,完成了当前任务流程怎么走,怎么生成新的任务都是在这里完成的。
进去看看:
- publicvoidsignal(StringsignalName,ObjectsignalData){
- ensureActivityInitialized();
- SignallableActivityBehavioractivityBehavior=(SignallableActivityBehavior)activity.getActivityBehavior();
- try{
- activityBehavior.signal(this,signalName,signalData);
- }catch(RuntimeExceptione){
- throwe;
- }catch(Exceptione){
- thrownewPvmException("couldn'tprocesssignal'"+signalName+"'onactivity'"+activity.getId()+"':"+e.getMessage(),e);
- }
- }
ExecutionEntity.signal方法核心工作就是把发信号的工作委托给PvmActivity的activityBehavior. 这里的设计存在问题,很显然其触犯了一个代码的坏味道:消息链。它让ExceutionEntity没有必要地与SignallableActivityBehavior 产生了耦合,更好的做法应该是PvmActivity提供signal方法,其内部调用ActivityBehavior完成发信号工作。
其实看看PvmActivity的接口声明,我不免也有疑问,本来属于PvmActivity的很重要的职责在其接口声明中都没有体现,why??
- publicinterfacePvmActivityextendsPvmScope{
- booleanisAsync();
- PvmScopegetParent();
- List<PvmTransition>getIncomingTransitions();
- List<PvmTransition>getOutgoingTransitions();
- PvmTransitionfindOutgoingTransition(StringtransitionId);
- }
把思路拉回来,我们继续看activityBehavior.signal方法内部的具体实现。
//待续
相关推荐
在标题和描述中提到的"activiti6.0.0源码",指的是该版本的Activiti引擎的原始代码,允许开发者深入理解其内部机制,并进行定制化开发或者修复潜在问题。 在深入探讨Activiti 6.0.0的知识点之前,我们先了解一下...
这个源码包是 Activiti 5.22.0 版本,包含了整个框架的源代码,为开发者提供了深入理解 Activiti 内部机制的机会。以下是一些关于 Activiti 的核心知识点: 1. **工作流程定义(BPMN 2.0)**:Activiti 支持 ...
在深入探讨Activiti7.0源码之前,我们先来了解一下BPM和Activiti的基本概念。 业务流程管理(Business Process Management,BPM)是一种用于优化企业运营的方法论,它关注于流程的设计、执行、监控和改进。BPM系统...
源码分析可以帮助我们深入理解Activiti的工作原理,包括任务调度、流程实例的创建与管理、活动的执行、信号与事件的处理等。通过查看源码,我们可以学习如何自定义行为、扩展Activiti的功能,以及如何进行性能优化。...
这个源码包是Activiti 5.22.0版本,是该框架的一个稳定版本,提供了完整的流程定义、执行、监控和集成能力。下面我们将深入探讨Activiti的核心特性、工作原理以及如何在Java项目中应用。 一、Activiti概述 ...
在这个"工作流activiti源码案例"中,我们可以深入学习如何在实际项目中应用Activiti。 1. Activiti概述:Activiti是基于BPMN 2.0标准的轻量级工作流引擎,它支持业务流程建模、执行和监控。BPMN(Business Process ...
Activiti 源码分析,流程文件部署主要涉及到 3 个表,分别是:ACT_GE_BYTEARRAY、ACT_RE_DEPLOYMENT、 ACT_RE_PROCDEF。主要完成“部署包”-->“流程定义文件”-->“所有包内文件”的解析部署关系 流程定义的部署...
Activiti是一个项目的名称,Alfresco软件在2010年5月17日宣布Activiti业务流程管理(BPM)开源项目的正式启动,其首席架构师由业务流程管理BPM的专家 Tom Baeyens担任。
这个资源是Activiti 7.0.45的源码,意味着我们可以深入理解其内部机制,对流程管理、任务调度等核心功能进行定制和扩展。 在Activiti 7.0.45的源码中,我们可以找到以下几个关键的知识点: 1. **Activiti架构**:...
Activiti 是一个开源的工作流和业务自动化引擎,它在Java平台上运行,并且与Spring框架有很好的集成。...对于想要为Activiti贡献代码或者开发基于Activiti的应用的人来说,这份源码无疑是无价的参考资料。
《SpringBoot集成Activiti项目Demo源码解析》 在当今的软件开发中,工作流引擎已经成为企业级应用不可或缺的一部分,而Activiti作为一款强大的、开源的工作流引擎,被广泛应用于业务流程管理(BPM)系统。...
这个5.22.0版本的源码提供了深入了解Activiti内部工作原理的机会,这对于开发者来说是宝贵的资源,特别是对于想要定制或者扩展Activiti功能的人员。 首先,让我们来看看`META-INF`目录。在Java应用中,`META-INF`...
这个"activiti源码-最新-中文注释"的压缩包提供了Activiti的源代码,并且有中文注释,这对于理解和学习Activiti的内部工作原理非常有帮助。 首先,让我们深入了解Activiti的基本概念。Activiti 是基于模型驱动的,...
在深入理解Activiti Designer 5.15.0的源码之前,我们需要对以下几个关键知识点有所了解: 1. **Activiti工作流引擎**:Activiti是一个开源的业务流程管理(BPM)和工作流引擎,用于在Java和Spring环境中构建企业级...
Open Source Business Automation Helping businesses solve automation challenges in distributed, highly-scalable and cost effective infrastructures.
本资源提供的"activiti web 例子 源码"是一个基于Web的Activiti实例,适用于开发者学习和理解如何在Web环境中集成与操作Activiti工作流引擎。 在Activiti 5版本中,它引入了许多关键特性,如: 1. **流程定义...
Activiti6最大的变化点就是对代码进行了重构,该版本修复以往的Bug并不多,但内部实现相对来说变化比较大。其突出的变化如下所示: 新增两款新引擎,Form引擎和DMN引擎(动态引擎)。其中DMN引擎允许开发人员创建...
在深入分析 Activiti 源码之前,我们先了解一下 Activiti 的基本概念。ExecutionEntity 是 Activiti 中的核心实体类,代表了流程实例的执行路径。它包含了一个 parent 属性,表示执行树结构,通过这个属性可以追踪...