activiti学习资料(架构描述)

Activiti学习资料

Activiti是业界很流行的java工作流引擎,关于Activiti与JBPM5的关系和如何选择不是本文要讨论的话题,相关内容可以baidu一下。Activiti从架构角度看是比较优秀的,是很面向对象的,是我所阅读过的代码结构很棒的开源软件,个人认为比Spring,Hibernate的要好。

Activiti的基础编程框架

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的关系,可以看下面这张图:

3.CommandContextIntercepter(CommandExecutor):Activiti使用命令模式作为基础开发模式,上面Service中定义的各个方法都对应相应的命令对象(xxCmd), Service把各种请求委托给xxCmd,xxCmd来决定命令的接收者,接收者执行后返回结果。而CommandContextIntercepter顾名思义,它是一个拦截器,拦截所有命令,在命令执行前后执行一些公共性操作。比如CommandContextIntercepter的核心方法:

[java] view plain copy

  1. public   T execute(Command  command) {
  2. CommandContext context = commandContextFactory.createCommandContext(command);
  3. try {
  4. //执行前保存上下文
  5. Context.setCommandContext(context);
  6. Context.setProcessEngineConfiguration(processEngineConfiguration);
  7. return next.execute(command);//执行命令
  8. } catch (Exception e) {
  9. context.exception(e);
  10. } finally {
  11. try {
  12. //关闭上下文,内部会flush session,把数据持久化到db等
  13. context.close();
  14. } finally {
  15. //释放上下文
  16. Context.removeCommandContext();
  17. Context.removeProcessEngineConfiguration();
  18. }
  19. }
  20. returnnull;
  21. }

关于命令模式的细节说明,网上有很多资料,这里不展开。我只是想说一下我看到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类图如下:

(图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包下)保持如下三类信息:

(图3:Context类)

CommandContext:命令上下文,保持每个命令需要的必要资源,如持久化需要的session。

ProcessEngineConfigurationImpl:流程引擎相关的配置信息。它是整个引擎的全局配置信息,mailServerHost,DataSource等。单例。该实例在流程引擎创建时被实例化,其调用stack如下图:

(图4:ProcessEngineConfiguration的初始化)

ExecutionContext:刚看到这个类感觉有些奇怪,不明白其作用是什么。看其代码持有ExecutionEntity这个实例。而ExecutionEntity是Activiti中非常重要的一个类,//TODO

6.Activiti的持久化框架(组件)

Activiti使用ibatis作为ORMapping工具。在此基础之上Activiti设计了自己的持久化框架,看一张图:

(图5:Activiti持久化框架)

顶层接口是Session和SessionFactory,这都非常好理解了。

Session有两个实现类:

DbSqlSession:简单点说,DbSqlSession负责sql表达式的执行。

AbstractManager:简单点说,AbstractManager及其子类负责面向对象的持久化操作

同理DbSqlSessionFactory与GenericManagerFactory的区别就很好理解了。

持久化框架也是在流程引擎建立时初始化的,具体见图4.

7.Event-Listener 组件

Activiti允许客户端代码介入流程的执行。为此提供了一个基础组件,看图:

(图6:用户代码介入流程的基础组件)

用户可以介入的代码类型包括:TaskListener,JavaDelegate,Expression,ExecutionListener。

ProcessEngineConfigurationImpl持有DelegateInterceptor的某个实例,这样就可以随时非常方便地调用handleInvocation

8.Cache 组件

对Activiti的cache实现很感兴趣,但现在我了解到的情况(也许还没有了解清楚)其cache的实现还是很简单的,在DbSqlSession中有cache实现:

[java] view plain copy

  1. protected List  insertedObjects =  new ArrayList ();
  2. protected Map , Map <string, cachedobject>> cachedObjects =  new HashMap , Map <string,cachedobject>>();
  3. protected List  deletedObjects =  new ArrayList ();
  4. protected List  deserializedObjects =  new ArrayList ();

也就是说Activiti就是基于内存的List和Map来做缓存的。具体怎么用的呢?以DbSqlSession.selectOne方法为例:

[java] view plain copy

  1. public Object selectOne(String statement, Object parameter) {
  2. statement = dbSqlSessionFactory.mapStatement(statement);
  3. Object result = sqlSession.selectOne(statement, parameter);
  4. if (result instanceof PersistentObject) {
  5. PersistentObject loadedObject = (PersistentObject) result;
  6. 缓存处理
  7. result = cacheFilter(loadedObject);
  8. }
  9. return result;
  10. }

[java] view plain copy

  1. protected PersistentObject cacheFilter(PersistentObject persistentObject) {
  2. PersistentObject cachedPersistentObject = cacheGet(persistentObject.getClass(), persistentObject.getId());
  3. if (cachedPersistentObject!=null) {
  4. //如果缓存中有就直接返回
  5. return cachedPersistentObject;
  6. }
  7. //否则,就先放入缓存
  8. cachePut(persistentObject, true);
  9. return persistentObject;
  10. }

[java] view plain copy

  1. protected CachedObject cachePut(PersistentObject persistentObject, boolean storeState) {
  2. Map <string, cachedobject> classCache = cachedObjects.get(persistentObject.getClass());
  3. if (classCache==null) {
  4. classCache = new HashMap <string, cachedobject>();
  5. cachedObjects.put(persistentObject.getClass(), classCache);
  6. }
  7. //这里是关键:一个CachedObject包含被缓存的对象本身:persistentObject和缓存的状态:storeState
  8. //Activiti正是根据storeState来判别缓存中的数据是否被更新是否与db保持一致的。
  9. CachedObject cachedObject = new CachedObject(persistentObject, storeState);
  10. classCache.put(persistentObject.getId(), cachedObject);
  11. return cachedObject;
  12. }

看了Activiti的缓存设计,我现在最大的疑问是Activiti貌似不支持cluster,因为其缓存设计是基于单机内存的,这个问题还需要进一步调查。

9.异步执行组件

Activiti可以异步执行job(具体例子可以看一下ProcessInstance startProcessInstanceByKey(String processDefinitionKey);),下面简单分析一下其实现过程,还是先看图:

(图7:异步执行组件核心类)

JobExecutor是异步执行组件的核心类,其包含三个主要属性:

1)JobAcquisitionThread jobAccquisitionThread:执行任务的线程 extends java.lang.Thread

2)BlockingQueue 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起来,例如:

[java] view plain copy

  1. PvmProcessDefinition processDefinition = new ProcessDefinitionBuilder()
  2. .createActivity("a").initial(). "background-color: #ff0000;" >behavior( new  WaitState())    
  3. .transition("b").endActivity().createActivity("b")
  4. .behavior(new WaitState()).transition("c").endActivity()
  5. .createActivity("c").behavior(new WaitState()).endActivity()
  6. .buildProcessDefinition();
  7. PvmProcessInstance processInstance = processDefinition
  8. .createProcessInstance();
  9. processInstance.start();
  10. PvmExecution activityInstance = processInstance.findExecution("a");
  11. assertNotNull(activityInstance);
  12. activityInstance.signal(null, null);
  13. activityInstance = processInstance.findExecution("b");
  14. assertNotNull(activityInstance);
  15. activityInstance.signal(null, null);
  16. activityInstance = processInstance.findExecution("c");
  17. assertNotNull(activityInstance);

以上代码都很简单,很好理解,只有一点需要说明一下,粗体红色背景的behavior方法,为一个PvmActivity增加ActivityBehavior,这是干什么呢?ActivityBehavior是一个interface,其接口声明很简单:

[java] view plain copy

  1. /**
  2. * @author Tom Baeyens
  3. */
  4. publicinterface ActivityBehavior {
  5. void execute(ActivityExecution execution) throws Exception;
  6. }

我的理解:Activiti把完成一个PvmActivity的行为单独建模封装在ActivityBehavior中。execute方法只有一个参数ActivityExecution,为啥这么设计?

为了更好地理解ActivityBehavior的作用,我们以TaskEntity.complete方法为例,分析其执行过程,先看complete的代码:

[java] view plain copy

  1. publicvoid complete() {
  2. fireEvent(TaskListener.EVENTNAME_COMPLETE);
  3. Context
  4. .getCommandContext()
  5. .getTaskManager()
  6. .deleteTask(this, TaskEntity.DELETE_REASON_COMPLETED, false);
  7. if (executionId!=null) {
  8. getExecution().signal(null, null);
  9. }
  10. }

代码很简单,也很好理解(可能出乎我们的意料,因为完成一个task,其实有很多事情要做的):

1.fireEvent:通知Listener,本任务完成了。

2.数据持久化相关的动作

3.getExecution().signal(null, null):发信号,这里面隐藏的东西就多了,总体来说,完成了当前任务流程怎么走,怎么生成新的任务都是在这里完成的。

进去看看:

[java] view plain copy

  1. publicvoid signal(String signalName, Object signalData) {
  2. ensureActivityInitialized();
  3. SignallableActivityBehavior activityBehavior = (SignallableActivityBehavior) activity.getActivityBehavior();
  4. try {
  5. activityBehavior.signal(this, signalName, signalData);
  6. } catch (RuntimeException e) {
  7. throw e;
  8. } catch (Exception e) {
  9. thrownew PvmException("couldn‘t process signal ‘"+signalName+"‘ on activity ‘"+activity.getId()+"‘: "+e.getMessage(), e);
  10. }
  11. }  "background-color: rgb(250, 250, 250); font-size: 1em; font-family: Monaco, ‘DejaVu Sans Mono‘, ‘Bitstream Vera Sans Mono‘, Consolas, ‘Courier New‘, monospace; ">

ExecutionEntity.signal方法核心工作就是把发信号的工作委托给PvmActivity的activityBehavior. 这里的设计存在问题,很显然其触犯了一个代码的坏味道:消息链。它让ExceutionEntity没有必要地与SignallableActivityBehavior 产生了耦合,更好的做法应该是PvmActivity提供signal方法,其内部调用ActivityBehavior完成发信号工作。

其实看看PvmActivity的接口声明,我不免也有疑问,本来属于PvmActivity的很重要的职责在其接口声明中都没有体现,why??

[java] view plain copy

  1. /**
  2. * @author Tom Baeyens
  3. */
  4. publicinterface PvmActivity extends PvmScope {
  5. boolean isAsync();
  6. PvmScope getParent();
  7. List  getIncomingTransitions();
  8. List  getOutgoingTransitions();
  9. PvmTransition findOutgoingTransition(String transitionId);
  10. }

把思路拉回来,我们继续看activityBehavior.signal方法内部的具体实现。

时间: 2024-08-25 22:47:46

activiti学习资料(架构描述)的相关文章

Activiti 学习资料收集

Activiti工作流引擎使用 http://www.open-open.com/lib/view/open1350460225367.html Activiti初学者教程 http://blog.csdn.net/bluejoe2000/article/details/39521405 基于Snaker的三种流程案例(选择分支,任务并行,会签) http://dead-knight.iteye.com/blog/1978302 http://www.open-open.com/doc/view

业余草分享100套精选1000G架构师资料课程(超1T的IT学习资料免费送)

业余草分享100套精选1000G架构师资料课程(超1T的IT学习资料免费送). 超过1024G的IT学习资料免费领取,你值得拥有! 领取资源方式,关注“业余草”公众号,回复对应的关键字 01.回复”我要健康“,获取程序员健康.内涵大礼包! 02.回复”大数据“,获取大数据相关资源. 03.回复”linux“,获取linux相关资料. 04.回复”mongodb“,获取mongodb学习资料. 05.回复”redis“,获取redis相关资料. 06.回复”zookeeper“,获取分布式等相关资

Java开发者的架构梦要如何实现(文末附送架构学习资料)

写在开篇不管是开发.测试.运维,每个技术人员心里都有一个成为技术大牛的梦,毕竟"梦想总是要有的,万一实现了呢"!正是对技术梦的追求,促使我们不断地努力和提升自己.然而"梦想是美好的,现实却是残酷的",很多同学在实际工作后就会发现,梦想是成为大牛,但做的事情看起来跟大牛都不沾边,例如,程序员说"天天写业务代码还加班,如何才能成为技术大牛",测试说"每天都有执行不完的测试用例",运维说"扛机器接网线敲shell命令,这不

优秀游戏程序员学习资料推荐

这两天给单位的技术做的一次学习材料推荐培训,直接ppt上拷过来的. 优秀游戏程序员学习资料推荐 主讲人:臧旭 前言 今天提到的纯粹是我个人心得和理解,可能片面,可能以偏概全. 目的是给大家做一定的指引作用,想让大家知道自己还有哪些可以去学习,还有哪些不足,我们距离优秀还有多远. 对我今天提到的东西,如果大家有时间,一定要去深入了解,在技术的道路上才有可能看得远.走得稳.飞得高. 另外有一句对所有技术人员想说的话: 学无止境.切忌坐井观天.有一点小小的成就就沾沾自喜.止足不前. 扎实的基础 万丈高

机器学习和深度学习资料合集

机器学习和深度学习资料合集 注:机器学习资料篇目一共500条,篇目二开始更新 希望转载的朋友,你可以不用联系我.但是一定要保留原文链接,因为这个项目还在继续也在不定期更新.希望看到文章的朋友能够学到更多.此外:某些资料在中国访问需要梯子. <Brief History of Machine Learning> 介绍:这是一篇介绍机器学习历史的文章,介绍很全面,从感知机.神经网络.决策树.SVM.Adaboost到随机森林.Deep Learning. <Deep Learning in

[转]机器学习和深度学习资料汇总【01】

本文转自:http://blog.csdn.net/sinat_34707539/article/details/52105681 <Brief History of Machine Learning> 介绍:这是一篇介绍机器学习历史的文章,介绍很全面,从感知机.神经网络.决策树.SVM.Adaboost到随机森林.Deep Learning. <Deep Learning in Neural Networks: An Overview> 介绍:这是瑞士人工智能实验室Jurgen

机器学习与深度学习资料

<Brief History of Machine Learning> 介绍:这是一篇介绍机器学习历史的文章,介绍很全面,从感知机.神经网络.决策树.SVM.Adaboost到随机森林.Deep Learning. <Deep Learning in Neural Networks: An Overview> 介绍:这是瑞士人工智能实验室Jurgen Schmidhuber写的最新版本<神经网络与深度学习综述>本综述的特点是以时间排序,从1940年开始讲起,到60-80

ABP 教程文档 1-1 手把手引进门之 AngularJs, ASP.NET MVC, Web API 和 EntityFramework(官方教程翻译版 版本3.2.5)含学习资料

本文是ABP官方文档翻译版,翻译基于 3.2.5 版本 转载请注明出处:http://www.cnblogs.com/yabu007/  谢谢 官方文档分四部分 一. 教程文档 二.ABP 框架 三.zero 模块 四.其他(中文翻译资源) 本篇是第一部分的第一篇. 第一部分分三篇 1-1 手把手引进门 1-2 进阶 1-3 杂项 (相关理论知识) 第一篇含两个步骤. 1-1-1 ASP.NET Core & Entity Framework Core 后端(内核)含两篇 ( 第一篇链接    

nodejs学习资料

NodeJS基础 什么是NodeJS JS是脚本语言,脚本语言都需要一个解析器才能运行.对于写在HTML页面里的JS,浏览器充当了解析器的角色.而对于需要独立运行的JS,NodeJS就是一个解析器. 每一种解析器都是一个运行环境,不但允许JS定义各种数据结构,进行各种计算,还允许JS使用运行环境提供的内置对象和方法做一些事情.例如运行在浏览器中的JS的用途是操作DOM,浏览器就提供了document之类的内置对象.而运行在NodeJS中的JS的用途是操作磁盘文件或搭建HTTP服务器,NodeJS