版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
1、 HYPERLINK http:/ Activiti源码分析Activiti的基础编程框架Activiti基于Spring,ibatis等开源中间件作为软件平台,在此之上构建了非常清晰的开发框架。上图列出了Activiti的核心组件。1.ProcessEngine:流程引擎的抽象,对于开发者来说,它是我们使用Activiti的facade,通过它可以获得我们需要的一切服务。2.XXService(TaskService,RuntimeService,RepositoryService.):Activiti按照流程的生命周期(定义,部署,运行)把不同阶段的服务封装在不同的Service中,用户可
2、以非常清晰地使用特定阶段的接口。通过ProcessEngine能够获得这些Service实例。TaskService,RuntimeService,RepositoryService是非常重要的三个Service:TaskService:流程运行过程中,与每个任务节点相关的接口,比如complete,delete,delegate等等RepositoryService:流程定义和部署相关的存储服务。RuntimeService:流程运行时相关服务,如startProcessInstanceByKey.关于ProcessEngine和XXService的关系,可以看下面这张图:3.Command
3、ContextIntercepter(CommandExecutor):Activiti使用命令模式作为基础开发模式,上面Service中定义的各个方法都对应相应的命令对象(xxCmd), Service把各种请求委托给xxCmd,xxCmd来决定命令的接收者,接收者执行后返回结果。而CommandContextIntercepter顾名思义,它是一个拦截器,拦截所有命令,在命令执行前后执行一些公共性操作。比如CommandContextIntercepter的核心方法:Java代码publicTexecute(Commandcommand)CommandContextcontext=comm
4、andContextFactory.createCommandContext(command);try/执行前保存上下文Context.setCommandContext(context);Context.setProcessEngineConfiguration(processEngineConfiguration);returnnext.execute(command);/执行命令catch(Exceptione)context.exception(e);finallytry/关闭上下文,内部会flushsession,把数据持久化到db等context.close();finally/释
5、放上下文Context.removeCommandContext();Context.removeProcessEngineConfiguration();returnnull;关于命令模式的细节说明,网上有很多资料,这里不展开。我只是想说一下我看到Activiti的这种设计之后的两点感受:1)一个产品或者一个项目,从技术上必须有一个明确的、唯一的开发模型或者叫开发样式(真不知道怎么说恰当),我们常说希望一个团队的所有人写出的代码都有统一的风格,都像是一个人写出来的,很理想化,但做到很难,往往我们都是通过“规范”去约束大家这样做,而规范毕竟是程序之外的东西,主观性很强,不遵守规范的情况屡屡发生
6、。而如果架构师给出了明确的开发模型,并使用一些基础组件加以强化,把程序员要走的路规定清楚,那你想不遵守规范都会很难,因为那意味着你写的东西没发工作。就像Activiti做的这样,明确以Command作为基本开发模型,辅之以Event-Listener,这样编程风格的整体性得到了保证。2)使用命令模式的好处,我这里体会最深的就是 职责分离,解耦。有了Command,各个Service从角色上说只是一些协调者或者控制者,他不需要知道具体怎么做,他只是把任务交给了各个命令。直接的好处是臃肿的、万能的大类没有了。而这往往是我们平时开发中最深恶痛绝的地方。4.核心业务对象(Task,ProcessIns
7、tance,Execution.):org.activiti.engine.impl.persistence.entity包下的类是Activiti的核心业务对象。它们是真正的对象,而不是只有数据没有行为的假对象,搞java企业级开发的人也许已经习惯了下面的层次划分:controller-service-dao-entity, entity只是ORMapping中数据表的java对象体现,没有任何行为(getter/setter不能算行为),对于面向对象来说,这当然是有问题的,记得曾听人说过这样的话“使用面向对象语言进行设计和开发 与 面向对象的设计和开发 是两回事”,面向对象讲究的是“封装”
8、,“多态”,追求的是满足“开-闭”原则的、职责单一的对象社会。如果你认同上述观点,那么相信Activiti会让你感觉舒服一些,以TaskEntity为例,其UML类图如下:(图2:TaskEntity类)TaskEntity实现了3个接口:Task,DelegateTask和PersistentObject。其中PersistentObject是一个声明接口,表明TaskEntity需要持久化。接口是一种角色的声明,是一份职责的描述而TaskEntity就是这个角色的具体扮演者,因此TaskEntity必须承担如complete,delegate等职责。但是这里有些遗憾的是像complete这
9、么重要的行为没有在3个接口中描述,因此“面向抽象”编程对于TaskEntity来说还没有完全做到。但至少Activiti告诉我们:1)牢记面向抽象编程,把职责拆分为不同的接口,让接口来体现对象的职责,而不用去关心这份职责具体由哪个对象实现;2)entity其实可以也应该是真正的对象。5.Activiti的上下文组件(Context)上下文(Context)用来保存生命周期很长的、全局性的信息。Activiti的Context类(在org.activiti.engine.impl.context包下)保持如下三类信息:(图3:Context类)CommandContext:命令上下文,保持每个命
10、令需要的必要资源,如持久化需要的session。ProcessEngineConfigurationImpl:流程引擎相关的配置信息。它是整个引擎的全局配置信息,mailServerHost,DataSource等。单例。该实例在流程引擎创建时被实例化,其调用stack如下图:(图4:ProcessEngineConfiguration的初始化)ExecutionContext:刚看到这个类感觉有些奇怪,不明白其作用。看其代码持有ExecutionEntity这个实例。而ExecutionEntity是Activiti中非常重要的一个类,/TODO6.Activiti的持久化框架(组件)Act
11、iviti使用ibatis作为ORMapping工具。在此基础之上Activiti设计了自己的持久化框架,看一张图:(图5:Activiti持久化框架)顶层接口是Session和SessionFactory,这都非常好理解了。Session有两个实现类:DbSqlSession:简单点说,DbSqlSession负责sql表达式的执行。AbstractManager:简单点说,AbstractManager及其子类负责面向对象的持久化操作同理DbSqlSessionFactory与GenericManagerFactory的区别就很好理解了。持久化框架也是在流程引擎建立时初始化的,具体见图4.
12、7.Event-Listener 组件Activiti允许客户端代码介入流程的执行。为此提供了一个基础组件,看图:(图6:用户代码介入流程的基础组件)用户可以介入的代码类型包括:TaskListener,JavaDelegate,Expression,ExecutionListener。ProcessEngineConfigurationImpl持有DelegateInterceptor的某个实例,这样就可以随时非常方便地调用handleInvocation8.Cache 组件对Activiti的cache实现很感兴趣,但现在我了解到的情况(也许还没有了解清楚)其cache的实现还是很简单的,
13、在DbSqlSession中有cache实现:Java代码protectedListinsertedObjects=newArrayList();protectedMapClass,MapcachedObjects=newHashMapClass,Map();protectedListdeletedObjects=newArrayList();protectedListdeserializedObjects=newArrayList();也就是说Activiti就是基于内存的List和Map来做缓存的。具体怎么用的呢?以DbSqlSession.selectOne方法为例:Java代码publ
14、icObjectselectOne(Stringstatement,Objectparameter)statement=dbSqlSessionFactory.mapStatement(statement);Objectresult=sqlSession.selectOne(statement,parameter);if(resultinstanceofPersistentObject)PersistentObjectloadedObject=(PersistentObject)result;缓存处理result=cacheFilter(loadedObject);returnresult;J
15、ava代码protectedPersistentObjectcacheFilter(PersistentObjectpersistentObject)PersistentObjectcachedPersistentObject=cacheGet(persistentObject.getClass(),persistentObject.getId();if(cachedPersistentObject!=null)/如果缓存中有就直接返回returncachedPersistentObject;/否则,就先放入缓存cachePut(persistentObject,true);returnper
16、sistentObject;Java代码protectedCachedObjectcachePut(PersistentObjectpersistentObject,booleanstoreState)MapclassCache=cachedObjects.get(persistentObject.getClass();if(classCache=null)classCache=newHashMap();cachedObjects.put(persistentObject.getClass(),classCache);/这里是关键:一个CachedObject包含被缓存的对象本身:persis
17、tentObject和缓存的状态:storeState/Activiti正是根据storeState来判别缓存中的数据是否被更新是否与db保持一致的。CachedObjectcachedObject=newCachedObject(persistentObject,storeState);classCache.put(persistentObject.getId(),cachedObject);returncachedObject;看了Activiti的缓存设计,我现在最大的疑问是Activiti貌似不支持cluster,因为其缓存设计是基于单机内存的,这个问题还需要进一步调查。9.异步执行组
18、件Activiti可以异步执行job(具体例子可以看一下ProcessInstance startProcessInstanceByKey(String processDefinitionKey);),下面简单分析一下其实现过程,还是先看图:(图7:异步执行组件核心类)JobExecutor是异步执行组件的核心类,其包含三个主要属性:1)JobAcquisitionThread jobAccquisitionThread:执行任务的线程 extends java.lang.Thread2)BlockingQueue threadPoolQueue3)ThreadPoolExecutor thr
19、eadPoolExecutor:线程池方法ProcessEngines在引擎启动时调用JobExecutor.start,JobAcquisitionThread 线程即开始工作,其run方法不断循环执行AcquiredJobs中的job,执行一次后线程等待一定时间直到超时或者JobExecutor.jobWasAdded方法因为有新任务而被调用。这里发现有一处设计的不够好:JobAcquisitionThread 与JobExecutor之间的关系是如此紧密(你中有我,我中有你),那么可以把JobAcquisitionThread 作为JobExecutor的内部类来实现,同时把Thread
20、PoolExecutor threadPoolExecutor交给JobAcquisitionThread 来管理,JobExecutor只负责接受任务以及启动、停止等更高级的工作,具体细节委托给JobAcquisitionThread ,责任分解,便于维护,JobExecutor的代码也会看起来更清晰。10.PVMPVM:Process Virtal Machine,流程虚拟机API暴露了流程虚拟机的POJO核心,流程虚拟机API描述了一个工作流流程必备的组件,这些组件包括:PvmProcessDefinition:流程的定义,形象点说就是用户画的那个图。静态含义。PvmProcessIns
21、tance:流程实例,用户发起的某个PvmProcessDefinition的一个实例,动态含义。PvmActivity:流程中的一个节点PvmTransition:衔接各个节点之间的路径,形象点说就是图中各个节点之间的连接线。PvmEvent:流程执行过程中触发的事件以上这些组件很好地对一个流程进行了建模和抽象。每个组件都有很清晰的角色和职责划分。另外,有了这些API,我们可以通过编程的方式,用代码来“画”一个流程图并让他run起来,例如:Java代码PvmProcessDefinitionprocessDefinition=newProcessDefinitionBuilder().cre
22、ateActivity(a).initial().behavior(newWaitState().transition(b).endActivity().createActivity(b).behavior(newWaitState().transition(c).endActivity().createActivity(c).behavior(newWaitState().endActivity().buildProcessDefinition();PvmProcessInstanceprocessInstance=processDefinition.createProcessInstanc
23、e();processInstance.start();PvmExecutionactivityInstance=processInstance.findExecution(a);assertNotNull(activityInstance);activityInstance.signal(null,null);activityInstance=processInstance.findExecution(b);assertNotNull(activityInstance);activityInstance.signal(null,null);activityInstance=processIn
24、stance.findExecution(c);assertNotNull(activityInstance);以上代码都很简单,很好理解,只有一点需要说明一下,粗体红色背景的behavior方法,为一个PvmActivity增加ActivityBehavior,这是干什么呢?ActivityBehavior是一个interface,其接口声明很简单:Java代码/*authorTomBaeyens*/publicinterfaceActivityBehaviorvoidexecute(ActivityExecutionexecution)throwsException;我的理解:Activi
25、ti把完成一个PvmActivity的行为单独建模封装在ActivityBehavior中。execute方法只有一个参数ActivityExecution,为啥这么设计?为了更好地理解ActivityBehavior的作用,我们以TaskEntplete方法为例,分析其执行过程,先看complete的代码:Java代码publicvoidcomplete()fireEvent(TaskListener.EVENTNAME_COMPLETE);Context.getCommandContext().getTaskManager().deleteTask(this,TaskEntity.DELE
26、TE_REASON_COMPLETED,false);if(executionId!=null)getExecution().signal(null,null);代码很简单,也很好理解(可能出乎我们的意料,因为完成一个task,其实有很多事情要做的):1.fireEvent:通知Listener,本任务完成了。2.数据持久化相关的动作3.getExecution().signal(null, null):发信号,这里面隐藏的东西就多了,总体来说,完成了当前任务流程怎么走,怎么生成新的任务都是在这里完成的。进去看看:Java代码publicvoidsignal(StringsignalName,
27、ObjectsignalData)ensureActivityInitialized();SignallableActivityBehavioractivityBehavior=(SignallableActivityBehavior)activity.getActivityBehavior();tryactivityBehavior.signal(this,signalName,signalData);catch(RuntimeExceptione)throwe;catch(Exceptione)thrownewPvmException(couldntprocesssignal+signal
28、Name+onactivity+activity.getId()+:+e.getMessage(),e);ExecutionEntity.signal方法核心工作就是把发信号的工作委托给PvmActivity的activityBehavior. 这里的设计存在问题,很显然其触犯了一个代码的坏味道:消息链。它让ExceutionEntity没有必要地与SignallableActivityBehavior 产生了耦合,更好的做法应该是PvmActivity提供signal方法,其内部调用ActivityBehavior完成发信号工作。其实看看PvmActivity的接口声明,我不免也有疑问,本来
29、属于PvmActivity的很重要的职责在其接口声明中都没有体现,why?Java代码/*authorTomBaeyens*/publicinterfacePvmActivityextendsPvmScopebooleanisAsync();PvmScopegetParent();ListgetIncomingTransitions();ListgetOutgoingTransitions();PvmTransitionfindOutgoingTransition(StringtransitionId);把思路拉回来,我们继续看activityBehavior.signal方法内部的具体实现。
30、/待续 HYPERLINK /shirdrn/article/details/6270506 Activiti 5.3:流程活动自动与手工触发执行分类: HYPERLINK /shirdrn/article/category/784507 Workflow2011-03-23 12:512849人阅读 HYPERLINK /shirdrn/article/details/6270506 l comments 评论(2) HYPERLINK javascript:void(0); o 收藏 收藏 HYPERLINK /shirdrn/article/details/6270506 l repor
31、t o 举报 举报 HYPERLINK /tag/details.html?tag=%e6%b4%bb%e5%8a%a8 t _blank 活动 HYPERLINK /tag/details.html?tag=exception t _blank exception HYPERLINK /tag/details.html?tag=class t _blank class HYPERLINK /tag/details.html?tag=string t _blank string HYPERLINK /tag/details.html?tag=object t _blank object HYP
32、ERLINK /tag/details.html?tag=deployment t _blank deploymentActiviti 5.3支持流程活动自动执行与手工触发执行。其中,自动执行是指,在启动流程之前,准备流程所需要的控制流程进度的变量数据,启动流程之后,无需外部干预,就能够按照预定义的流程执行;手工触发执行是指,执行到流程中某个个结点后流程暂时停止运行,直到收到外部发送的信号以后,才会继续向前推进,这样情况可以更加精细地控制流程。下面主要通过基于Activiti 5.3的、元素来看一下。首先,我们在测试的过程中,用到JUnit 3.x,为了方便,这里给了一层封装,代码如下所示:j
33、ava HYPERLINK /shirdrn/article/details/6270506 o view plain view plain HYPERLINK /shirdrn/article/details/6270506 o copy copypackageorg.shirdrn.workflow.activiti;importjunit.framework.TestCase;importorg.activiti.engine.FormService;importorg.activiti.engine.HistoryService;importorg.activiti.engine.Id
34、entityService;importorg.activiti.engine.ManagementService;importorg.activiti.engine.ProcessEngine;importorg.activiti.engine.ProcessEngines;importorg.activiti.engine.RepositoryService;importorg.activiti.engine.RuntimeService;importorg.activiti.engine.TaskService;/*authorshirdrn*/publicabstractclassAb
35、stractTestextendsTestCaseprivateProcessEngineprocessEngine;protectedStringdeploymentId;protectedRepositoryServicerepositoryService;protectedRuntimeServiceruntimeService;protectedTaskServicetaskService;protectedFormServiceformService;protectedHistoryServicehistoryService;protectedIdentityServiceident
36、ityService;protectedManagementServicemanagementService;OverrideprotectedvoidsetUp()throwsExceptionsuper.setUp();if(processEngine=null)processEngine=ProcessEngines.getDefaultProcessEngine();repositoryService=processEngine.getRepositoryService();runtimeService=processEngine.getRuntimeService();taskSer
37、vice=processEngine.getTaskService();formService=processEngine.getFormService();historyService=processEngine.getHistoryService();identityService=processEngine.getIdentityService();managementService=processEngine.getManagementService();initialize();OverrideprotectedvoidtearDown()throwsExceptionsuper.t
38、earDown();destroy();protectedabstractvoidinitialize()throwsException;protectedabstractvoiddestroy()throwsException;这里面,主要是在测试之前做一些初始化工作,主要包括流程引擎实例的构建,及其流程提供的基本服务。下面测试会用到该抽象类。自动执行元素,可以实现自动活动,语法如下所示:xhtml HYPERLINK /shirdrn/article/details/6270506 o view plain view plain HYPERLINK /shirdrn/article/det
39、ails/6270506 o copy copy其中,activiti:class属性为该结点对应的处理类,该类要求实现org.activiti.engine.delegate.JavaDelegate接口,该接口定义如下所示:java HYPERLINK /shirdrn/article/details/6270506 o view plain view plain HYPERLINK /shirdrn/article/details/6270506 o copy copypackageorg.activiti.engine.delegate;publicinterfaceJavaDeleg
40、atevoidexecute(DelegateExecutionexecution)throwsException;execute方法的参数DelegateExecution execution可以在流程中各个结点之间传递流程变量。下面给出一个具体的例子:自动执行的流程,如图所示:对应的流程定义文件为GatewayTest.testAutomaticForkJoin.bpmn20.xml,如下所示:c-sharp HYPERLINK /shirdrn/article/details/6270506 o view plain view plain HYPERLINK /shirdrn/artic
41、le/details/6270506 o copy copy上述流程定义中,一共定义了4个ServiceTask,模拟实现代码如下所示:java HYPERLINK /shirdrn/article/details/6270506 o view plain view plain HYPERLINK /shirdrn/article/details/6270506 o copy copypackageorg.shirdrn.workflow.activiti.gateway;importjava.util.logging.Logger;importorg.activiti.engine.dele
42、gate.DelegateExecution;importorg.activiti.engine.delegate.JavaDelegate;publicclassServiceTask1implementsJavaDelegateprivatefinalLoggerlog=Logger.getLogger(ServiceTask1.class.getName();Overridepublicvoidexecute(DelegateExecutionexecution)throwsExceptionThread.sleep(10000);(variavles=+execution.getVar
43、iables();execution.setVariable(task1,Iamtask1);(Iamtask1.);java HYPERLINK /shirdrn/article/details/6270506 o view plain view plain HYPERLINK /shirdrn/article/details/6270506 o copy copypackageorg.shirdrn.workflow.activiti.gateway;importjava.util.logging.Logger;publicclassServiceTask2implementsJavaDe
44、legateprivatefinalLoggerlog=Logger.getLogger(ServiceTask2.class.getName();Overridepublicvoidexecute(DelegateExecutionexecution)throwsExceptionThread.sleep(10000);(variavles=+execution.getVariables();execution.setVariable(task2,Iamtask2);(Iamtask2.);java HYPERLINK /shirdrn/article/details/6270506 o v
45、iew plain view plain HYPERLINK /shirdrn/article/details/6270506 o copy copypackageorg.shirdrn.workflow.activiti.gateway;importjava.util.logging.Logger;publicclassServiceTask3implementsJavaDelegateprivatefinalLoggerlog=Logger.getLogger(ServiceTask3.class.getName();Overridepublicvoidexecute(DelegateEx
46、ecutionexecution)throwsExceptionThread.sleep(10000);(variavles=+execution.getVariables();execution.setVariable(task3,Iamtask3);(Iamtask3.);java HYPERLINK /shirdrn/article/details/6270506 o view plain view plain HYPERLINK /shirdrn/article/details/6270506 o copy copypackageorg.shirdrn.workflow.activit
47、i.gateway;importjava.util.logging.Logger;publicclassServiceTask4implementsJavaDelegateprivatefinalLoggerlog=Logger.getLogger(ServiceTask4.class.getName();Overridepublicvoidexecute(DelegateExecutionexecution)throwsExceptionThread.sleep(10000);(variavles=+execution.getVariables();execution.setVariable
48、(task4,Iamtask4);(Iamtask4.);测试代码,如下所示:java HYPERLINK /shirdrn/article/details/6270506 o view plain view plain HYPERLINK /shirdrn/article/details/6270506 o copy copypackageorg.shirdrn.workflow.activiti.gateway;importorg.activiti.engine.runtime.ProcessInstance;importorg.activiti.engine.test.Deploymen
49、t;importorg.shirdrn.workflow.activiti.AbstractTest;/*authorshirdrn*/publicclassAutomaticParallelGatewayTestextendsAbstractTestprivateStringdeploymentId;Overrideprotectedvoidinitialize()throwsExceptiondeploymentId=repositoryService.createDeployment().addClasspathResource(diagrams/GatewayTest.testAuto
50、maticForkJoin.bpmn20.xml).deploy().getId();Overrideprotectedvoiddestroy()throwsExceptionrepositoryService.deleteDeployment(deploymentId,true);DeploymentpublicvoidtestForkJoin()ProcessInstancepi=runtimeService.startProcessInstanceByKey(AutomaticParalellBasedForkJoin);assertEquals(true,pi.isEnded();只需
51、要启动一个流程实例,它会自动执行到结束。这种情况下,你不需要关注流程的执行进度,而只需要把精力集中在每个结点的处理逻辑(通常是简单或者复杂的商业逻辑)上,运行结果如下所示:xhtml HYPERLINK /shirdrn/article/details/6270506 o view plain view plain HYPERLINK /shirdrn/article/details/6270506 o copy copy2011-3-2311:50:12org.shirdrn.workflow.activiti.gateway.ServiceTask1execute信息:variavles=
52、2011-3-2311:50:12org.shirdrn.workflow.activiti.gateway.ServiceTask1execute信息:Iamtask1.2011-3-2311:50:22org.shirdrn.workflow.activiti.gateway.ServiceTask2execute信息:variavles=task1=Iamtask12011-3-2311:50:22org.shirdrn.workflow.activiti.gateway.ServiceTask2execute信息:Iamtask2.2011-3-2311:50:32org.shirdr
53、n.workflow.activiti.gateway.ServiceTask4execute信息:variavles=task1=Iamtask1,task2=Iamtask22011-3-2311:50:32org.shirdrn.workflow.activiti.gateway.ServiceTask4execute信息:Iamtask4.2011-3-2311:50:42org.shirdrn.workflow.activiti.gateway.ServiceTask3execute信息:variavles=task1=Iamtask1,task2=Iamtask2,task4=Ia
54、mtask42011-3-2311:50:42org.shirdrn.workflow.activiti.gateway.ServiceTask3execute信息:Iamtask3.手工触发执行通过和元素都可以实现流程的手工触发执行。基于实现的流程,如图所示:对应的流程定义文件Task.ReceiveTask.bpmn20.xml,如下所示:xhtml HYPERLINK /shirdrn/article/details/6270506 o view plain view plain HYPERLINK /shirdrn/article/details/6270506 o copy copy
55、上述流程定义中,对应的两个处理类,代码分别如下所示:java HYPERLINK /shirdrn/article/details/6270506 o view plain view plain HYPERLINK /shirdrn/article/details/6270506 o copy copypackageorg.shirdrn.workflow.activiti.task;importjava.util.HashMap;importjava.util.logging.Logger;importorg.activiti.engine.delegate.DelegateExecutio
56、n;importorg.activiti.engine.delegate.JavaDelegate;publicclassCheckBankReceiveTaskimplementsJavaDelegateprivatefinalLoggerlog=Logger.getLogger(CheckBankReceiveTask.class.getName();SuppressWarnings(unchecked)Overridepublicvoidexecute(DelegateExecutionexecution)throwsException(iamCheckBankReceiveTask.)
57、;System.out.println(in:+execution.getVariables();(HashMap)execution.getVariables().get(in).put(next,CheckBankTask);(HashMap)execution.getVariables().get(out).put(reponse,subprocess:CheckBankReceiveTask-CheckMerchantReceiveTask);java HYPERLINK /shirdrn/article/details/6270506 o view plain view plain
58、HYPERLINK /shirdrn/article/details/6270506 o copy copypackageorg.shirdrn.workflow.activiti.task;importjava.util.HashMap;importjava.util.logging.Logger;importorg.activiti.engine.delegate.DelegateExecution;importorg.activiti.engine.delegate.JavaDelegate;publicclassCheckMerchantReceiveTaskimplementsJav
59、aDelegateprivatefinalLoggerlog=Logger.getLogger(CheckMerchantReceiveTask.class.getName();SuppressWarnings(unchecked)Overridepublicvoidexecute(DelegateExecutionexecution)throwsException(iamCheckMerchantReceiveTask.);System.out.println(in:+execution.getVariables();(HashMap)execution.getVariables().get
60、(in).put(previous,CheckMerchantReceiveTask);上面还用到一个org.shirdrn.workflow.activiti.subprocess.Merchant类,该类必须支持序列化,如下所示:java HYPERLINK /shirdrn/article/details/6270506 o view plain view plain HYPERLINK /shirdrn/article/details/6270506 o copy copypackageorg.shirdrn.workflow.activiti.subprocess;importjav
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- 临沂电商整治方案
- 空调营销方案
- 新疆行政职业能力测验模拟34
- 新课标理念下小学足球大单元整体设计的思考与实践
- 本土文化资源融入幼儿园游戏活动的路径
- 广东公务员面试模拟9
- 河南面试模拟48
- 贵州公务员面试模拟13
- 四川行政职业能力模拟14
- 四川省申论模拟205
- 《人工智能基础》课件-AI的前世今生:她从哪里来
- 3.2 代数式的值(第1课时)(课件)-2024-2025学年七年级数学上册(人教版2024)
- 秦腔传统剧《草坡面理》
- 直流电机设计参数计算
- 员工上岗认证管理办法
- 核心素养下小学语文教学策略探究
- 必备职业素养(PPT课件)
- 十以内加减法口算题
- 浅谈柔韧训练在中学生体能训练中的重要性(张磊
- 工程监理工作流程图大全WORD完整版
- 实验一蒸馏工业乙醇
评论
0/150
提交评论