简单状态机Workflow基于Web应用

出处:http://www.cnblogs.com/cxd4321/archive/2007/09/10/888006.html

一、实现目的
1、在Web中托管Workflow引擎实现工作流流转。
2
、使用持久化的方式存放工作流实例状态。
二、基本原形
报销流程(网上有很成熟的基于工作流的报销流程,我还在这里显摆,唉!)
三、使用到的核心类
WorkflowRuntimeExternalDataExchangeServiceWorkflowInstanceWorkflowPersistenceService
分别在dotnet3.0的一下命名空间中:
System.Workflow.Runtime
System.Workflow.ActivitiesSystem.Workflow.Runtime.Hosting
四、工作流项目设计

1、报销流程还是沿用了网上广为流传的那个报销流程的模式(细节上也没有太大的差别),这里简单描述一下:报销者提交报销信息——〉部门经理审批——〉副总经理审批(如果数目小于1000元跳过此环节)——〉财务经理审批——〉出纳确认——〉结束,如果在除提交环节的任何一个环节审批不通过则将打回到提交者进行提交的初始状态。该例子中使用了状态机模式(个人认为状态机的Workflow可以很好的满足我以前开发中的大多数需求)。
2
、使用VS2005进行Workflow开发的基本条件网上很多,这里简单的说一下:
dotnetfx3
Visual Studio 2005 Extensions for Windows Workflow Foundation (CHS)[都可以从官方下载]
3
、安装好后会在项目模板中出现专门为Workflow工程开发所提供的几个模板

简单状态机Workflow基于Web应用
我个人平时喜欢建立带有控制台的,那样调试很方便,当然如果那样做了的话后来还需要再创建一个基于库的,把那些代码再粘贴过去,因为控制台的bin文件夹里只有exe文件,不能让其他项目进行引用(不过可能有其他的方法我不知道)
4
、在工具箱中选择创建流程的各种活动Actives在本例中只适用到了五种StateEventDrivenHandleExternalEventSetSateIfElse
简单状态机Workflow基于Web应用

其中State就是状态机中的每个状态的最大容器(当然在设计工作流的时候也可以把一些Actives放在State之外,比如EventDriven,但我还不知道放在状态之外的用处),我设计的其他Actives都是在State里面的。其实我个人使用Active时那些能放在另外一些里面都是通过将他们托放看能不能实现来判断的。

5
、根据我的业务流程,在这个例子中总共使用到了6种状态,起始[InitState]和结束[EndState]好像是状态机工作流中必须的两个状态,而且好像能有多个起始状态,结束状态之有一个。此外在该例中还有部门经理审批状态[DManagerState]、副总审批状态[GManagerState]、财务经理审批状态[FManagerState]、出纳确认状态[CashierState](本人的状态分解可能在逻辑上不甚合理,请谅解)。每个状态中都有通过事件进行驱动的EventDriven活动(一个为审批通过触发的事件,一个为审批未通过),还有用来设置下一个进入某个状态的SetState活动。只有在部门经理审批状态中添加了判断报销金额再进行环节流转的Actives。包含关系大概为State活动中包含(拖放)EventDrive活动,EventDrive活动中包含(拖放)HandleExternalEventSetSateIfElse活动。
(一个状态图片)
简单状态机Workflow基于Web应用
(进入EventDriven的图片)
简单状态机Workflow基于Web应用
(部门经理审批通过触发的EventDriven中判断金额进行状态选择流转的图片)
简单状态机Workflow基于Web应用
6
、在工作流图形定制界面中除了将所有的Actives拖入放置好以外,还需要对每个Actives进行设置。几种类型Active的大致设置方法为:
State
EventDriven活动只需要进行名称的设置就可以了因为他们可能只是做其他事件或者操作方式的一个容器,不提供什么方法。
HandleExternalEvent
活动需要设置需要处理的外部事件,这些外部事件需要在一个定义了很多事件类型的接口来选择。其实就是我们需要定义一个接口,在接口中定义需要在HandleExternalEvent中调用的事件,然后再实现这个接口,在宿主环境中调用接口的实现类来对Workflow引擎发送事件消息,HandleExternalEvent事件在得到这个消息之前是等待状态,也就是说会引发workflowRuntimeWorkflowIdled事件(我们会在这里处理工作流持久化)。HandleExternalEvent得到这个事件消息之后就会往下执行了。因此我们需要设置HandleExternalEvent活动的一下两个属性值InterfaceTypeEventName,其中InterfaceType是我们定义的那个接口,EventName是我们在接口中定义的用来为特定HandleExternalEvent进行触发的事件。我们也可以通过使用Invoked来在工作流中处理一些需要的操作。
简单状态机Workflow基于Web应用

IfElse活动作为一个容器只需要设置名称,其中需要特别注意的是活动中所包含的两个分支,每个分支都可以绑定进行判断的方法。也可以只在左边的分支上进行条件判断,条件判断可以选择使用“Declarative Rule Condition”申明规则条件(本例中未使用,本人也不太清楚怎么用),还有一个“Code Condition”进行判断方法的绑定,本例中使用这种。选择后可以展开Condition属性将自己写好的条件方法代码绑定。

IfElse活动)
简单状态机Workflow基于Web应用

IfElse分支属性)
简单状态机Workflow基于Web应用

SetState活动只需要设置它的名称和TargetStateName属性,该属性的作用是选择下一步要流入的状态名称
简单状态机Workflow基于Web应用
7
、在工作流代码文件中需要完成IfElse活动中需要的条件判断方法和一个用来接收从宿主(hosting)环境中传入的用来进行报销金额是否大于1000元的参数属性。
在工作流设计的解决方案资源管理器中选择查看代码,加入以上代码,并最终绑定IfElse活动的Condition属性。

usingSystem;
usingSystem.ComponentModel;
usingSystem.ComponentModel.Design;
usingSystem.Collections;
usingSystem.Drawing;
usingSystem.Workflow.ComponentModel.Compiler;
usingSystem.Workflow.ComponentModel.Serialization;
usingSystem.Workflow.ComponentModel;
usingSystem.Workflow.ComponentModel.Design;
usingSystem.Workflow.Runtime;
usingSystem.Workflow.Activities;
usingSystem.Workflow.Activities.Rules;

namespaceWorkflow
{
publicsealedpartialclassWorkflow:StateMachineWorkflowActivity
{
//报销金额[以后有了Orcas就不需要定义这个变量了]
privatestring_strMoneyNum=default(string);

//报销金额[属性]
publicstringMoneyNum
{
get
{
return_strMoneyNum;
}
set
{
_strMoneyNum
=value;
}
}

publicWorkflow()
{
InitializeComponent();
}

///<summary>
///IfElse活动左分支的CodeCondition中绑定的方法
///</summary>
///<paramname="sender">发送对象</param>
///<paramname="e">注意这里的事件数据类型必须为ConditionalEventArgs</param>
publicvoidGManagerOrFManager(objectsender,ConditionalEventArgse)
{
//设置e中需要给工作流返回的bool参数
if(Convert.ToInt16(MoneyNum.Trim())>1000)
e.Result
=true;
else
e.Result
=false;
}

}

}

8、为每个需要接收事件的HandleExternalEvent活动定义一个统一的ExternalDataExchangeService类型接口,并且实现该接口,这样可以在宿主环境中使用接口实现类的实例同工作流引擎通信。
9
、在整个解决方案中创建新的Windows类库项目,本例中创建名称为ExtendServiceInstance的项目
10
、在解决方案中创建一个以[ExternalDataExchange]为属性的接口,本例中文件名称为ICharge

usingSystem;
usingSystem.Workflow.Activities;

namespaceExtendServiceInstance
{
[ExternalDataExchange]
publicinterfaceICharge
{
#regionICharge成员

eventEventHandler<ExternalDataEventArgs>OnInitStart;

eventEventHandler<ExternalDataEventArgs>OnDManagerStart;

eventEventHandler<ExternalDataEventArgs>OnDManagerRStart;

eventEventHandler<ExternalDataEventArgs>OnGManagerStart;

eventEventHandler<ExternalDataEventArgs>OnGManagerRStart;

eventEventHandler<ExternalDataEventArgs>OnFManagerStart;

eventEventHandler<ExternalDataEventArgs>OnFManagerRStart;

eventEventHandler<ExternalDataEventArgs>OnCashierStart;

eventEventHandler<ExternalDataEventArgs>OnCashierRStart;

eventEventHandler<ExternalDataEventArgs>OnEndStart;

#endregion
}
}


11、创建实现该接口的类,本例中文件名称为Charge

usingSystem;
usingSystem.Workflow.Activities;

namespaceExtendServiceInstance
{
[Serializable]
publicclassCharge:ICharge
{
#regionICharge成员

publiceventEventHandler<ExternalDataEventArgs>OnInitStart;

publiceventEventHandler<ExternalDataEventArgs>OnDManagerStart;

publiceventEventHandler<ExternalDataEventArgs>OnDManagerRStart;

publiceventEventHandler<ExternalDataEventArgs>OnGManagerStart;

publiceventEventHandler<ExternalDataEventArgs>OnGManagerRStart;

publiceventEventHandler<ExternalDataEventArgs>OnFManagerStart;

publiceventEventHandler<ExternalDataEventArgs>OnFManagerRStart;

publiceventEventHandler<ExternalDataEventArgs>OnCashierStart;

publiceventEventHandler<ExternalDataEventArgs>OnCashierRStart;

publiceventEventHandler<ExternalDataEventArgs>OnEndStart;

#endregion

publicvoidRaiseInitStartEvent(GuidinstanceId)
{
if(OnInitStart!=null)
{
OnInitStart(
this,(newExternalDataEventArgs(instanceId)));
}
}

publicvoidRaiseDManagerStartEvent(GuidinstanceId)
{
if(OnDManagerStart!=null)
{
OnDManagerStart(
this,(newExternalDataEventArgs(instanceId)));
}
}

publicvoidRaiseDManagerRStartEvent(GuidinstanceId)
{
if(OnDManagerRStart!=null)
{
OnDManagerRStart(
this,(newExternalDataEventArgs(instanceId)));
}
}

publicvoidRaiseGManagerStartEvent(GuidinstanceId)
{
if(OnGManagerStart!=null)
{
OnGManagerStart(
this,(newExternalDataEventArgs(instanceId)));
}
}

publicvoidRaiseGManagerRStartEvent(GuidinstanceId)
{
if(OnGManagerRStart!=null)
{
OnGManagerRStart(
this,(newExternalDataEventArgs(instanceId)));
}
}

publicvoidRaiseFManagerStartEvent(GuidinstanceId)
{
if(OnFManagerStart!=null)
{
OnFManagerStart(
this,(newExternalDataEventArgs(instanceId)));
}
}

publicvoidRaiseFManagerRStartEvent(GuidinstanceId)
{
if(OnFManagerRStart!=null)
{
OnFManagerRStart(
this,(newExternalDataEventArgs(instanceId)));
}
}

publicvoidRaiseCashierStartEvent(GuidinstanceId)
{
if(OnCashierStart!=null)
{
OnCashierStart(
this,(newExternalDataEventArgs(instanceId)));
}
}

publicvoidRaiseCashierRStartEvent(GuidinstanceId)
{
if(OnCashierRStart!=null)
{
OnCashierRStart(
this,(newExternalDataEventArgs(instanceId)));
}
}

publicvoidRaiseEndStartEvent(GuidinstanceId)
{
if(OnEndStart!=null)
{
OnEndStart(
this,(newExternalDataEventArgs(instanceId)));
}
}
}
}

类中的每个方法对应一个事件,目的是使用这些方法来触发所定义的事件进一步激发和工作流引擎的通信。12、在之前创建的工作流设计工程中引用我们刚才创建的类库项目
简单状态机Workflow基于Web应用

13、为之前在工作流设计中定义的所有HandleExternalEvent活动绑定接口和事件
14、编译工作流工程,如果没有编译错误,到这里工作流的设计就算完成了。
五、配置工作流持久化SQL Server数据库
本例中使用的工作流持久化方法是使用SqlWorkflowPersistenceService来对WorkflowPersistenceService服务进行实例化,因此需要在一台SQL Server数据库中配置实例化所需要的持久化库。本例以SQL Server 2005为例进行配置。
1
、配置数据库前确保<%WINDOWS%>\Microsoft.NET\Framework\v3.0\Windows Workflow Foundation\SQL\EN文件夹和其中的SqlPersistenceService_Logic.sql以及SqlPersistenceService_Schema.sql文件的存在。确保方法是安装dotnet3.0
2
、开发SQL Server 2005SQL Server Management Studio Express
简单状态机Workflow基于Web应用
3
、登陆后选择数据库里面的新建
简单状态机Workflow基于Web应用
4
、数据库名称可自定,在本例中名称为SqlPersistenceService
5
、选择文件”——打开”——文件
简单状态机Workflow基于Web应用
6、选择咱们刚才要求确保的那个目录中的sql脚本文件进行执行,顺序为先SqlPersistenceService_Schema.sqlSqlPersistenceService_Logic.sql。以SqlPersistenceService_Schema.sql为例演示:

打开SqlPersistenceService_Schema.sql文件后,需要再次登陆数据库。登陆完成后,在下拉框中选择咱们刚才新建的数据库

简单状态机Workflow基于Web应用
点击!执行
SqlPersistenceService_Logic.sql
的执行过程同上。
7
、至此持久化数据库配置完成。
六、宿主环境设计
1、在解决方案中新添加一个Web工程,本例中工程名称为WebUI,在工程中添加四个aspx页面进行项目运行时的宿主和跳转。
Default.aspx
加载报销条目的GridView页面
Info.aspx
工作流的宿主页面
Insert.aspx
添加报销信息页面
Login.aspx
登陆界面
2
、因为本例核心为工作流的应用,所以我只介绍Info.aspx界面中有关宿主的核心内容。
3
、在工程中添加引用将我们前边创建的工作流工程引用进来。
简单状态机Workflow基于Web应用
4
、在Info.aspx文件的using 中需要加入以下几个引用。

usingSystem.Workflow.Runtime;
usingSystem.Workflow.Runtime.Hosting;
usingSystem.Workflow.Activities;
usingSystem.Collections.Generic;
usingExtendServiceInstance;
usingWorkflow;

之后要使用到的WorkflowRuntimeWorkflowInstance System.Workflow.Runtime命名空间下,ChargeExtendServiceInstance命名空间下,ExternalDataExchangeServiceSystem.Workflow.Activities命名空间下,WorkflowPersistenceServiceSqlWorkflowPersistenceServiceSystem.Workflow.Runtime.Hosting命名空间下,DictionarySystem.Collections.Generic命名空间下
5
、使用状态机工作流的大致步骤为:
实例化工作流引擎 WorkflowRuntime——〉绑定工作流引擎中的事件——〉加载各种需要的服务——〉启动工作流引擎——〉如果是新创建工作流实例则是用WorkflowRuntime .CreateWorkflow方法,如果是从持久化SQL Server数据库中恢复工作流实例则使用workflowRuntime.GetWorkflowWorkflowInstance.Load方法——〉使用StateMachineWorkflowInstance强制转化工作流实例来获得状态机工作流所有状态或者当前状态等——〉在WorkflowRuntimeWorkflowIdled事件所绑定的方法中调用e.WorkflowInstance.TryUnload()方法进行工作流向SQL Server数据库的写入。
6、以下是Info.aspx中核心的几段带注释的代码

实例化

///<summary>
///定义工作流引擎对象,加载各种服务类、工作流实例或者创建工作流实例
///</summary>
privatestaticWorkflowRuntimeworkflowRuntime=default(WorkflowRuntime);
///<summary>
///定义Charge服务类,用来和工作流实例通讯
///</summary>
privatestaticChargechargeService=default(Charge);
///<summary>
///定义工作流实例
///</summary>
privatestaticWorkflowInstanceinstance=default(WorkflowInstance);
///<summary>
///定义扩展服务对象用来加载Charge服务类,并被工作流引擎加载
///</summary>
privatestaticExternalDataExchangeServicedataExchangeService=default(ExternalDataExchangeService);
///<summary>
///定义持久化服务
///</summary>
privatestaticWorkflowPersistenceServicepersistenceService=default(WorkflowPersistenceService);

工作流引擎初始化、绑定服务、最后启动

if(workflowRuntime==null)
workflowRuntime
=newWorkflowRuntime();
if(chargeService==null)
chargeService
=newCharge();
if(persistenceService==null)
//实例化SqlWorkflowPersistenceService
persistenceService=
newSqlWorkflowPersistenceService(
"DataSource=10.18.100.31;InitialCatalog=SqlPersistenceService;PersistSecurityInfo=True;UserID=sa;Password=skyadmin;",
false,
newTimeSpan(1,0,0),
newTimeSpan(0,0,5));
if(workflowRuntime.GetService(persistenceService.GetType())==null)
//绑定服务
workflowRuntime.AddService(persistenceService);

//对工作流引擎中的各种事件进行需要的绑定
workflowRuntime.WorkflowCompleted+=OnWorkflowCompleted;
workflowRuntime.WorkflowIdled
+=OnWorkflowIdled;
workflowRuntime.WorkflowPersisted
+=OnWorkflowPersisted;
workflowRuntime.WorkflowUnloaded
+=OnWorkflowUnloaded;
workflowRuntime.WorkflowLoaded
+=OnWorkflowLoaded;
workflowRuntime.WorkflowTerminated
+=OnWorkflowTerminated;
workflowRuntime.WorkflowAborted
+=OnWorkflowAborted;

//绑定服务
dataExchangeService=newExternalDataExchangeService();
if(workflowRuntime.GetService(dataExchangeService.GetType())==null)
{
workflowRuntime.AddService(dataExchangeService);
if(dataExchangeService.GetService(chargeService.GetType())==null)
dataExchangeService.AddService(chargeService);
}
//启动引擎
workflowRuntime.StartRuntime();


新建工作流实例

//使用Dictionary传递参数
Dictionary<string,object>parameters=newDictionary<string,object>();
parameters.Add(
"MoneyNum",oleDS.Tables[0].Rows[0]["AC_MONEY"].ToString().Trim());
//创建工作流实例
instance=workflowRuntime.CreateWorkflow(typeof(Workflow.Workflow),parameters);
//工作流实例启动
instance.Start();


从持久化数据库加载工作流实例

//工作流引擎被唤醒后加载工作流实例
instance=workflowRuntime.GetWorkflow(newGuid(oleDS.Tables[0].Rows[0]["AC_WORKFLOWID"].ToString().Trim()));
instance.Load();


工作流引擎挂起时进行持久化设置

privatevoidOnWorkflowIdled(objectsender,WorkflowEventArgse)
{
//调用SqlWorkflowPersistenceService进行持久化
e.WorkflowInstance.TryUnload();
}


工作流完成时进行自己的操作

简单状态机Workflow基于Web应用privatevoidOnWorkflowCompleted(objectsender,WorkflowCompletedEventArgsinstance)
简单状态机Workflow基于Web应用简单状态机Workflow基于Web应用
简单状态机Workflow基于Web应用{
简单状态机Workflow基于Web应用
//添加工作流完成后需要的操作
简单状态机Workflow基于Web应用
}


七、设计中需要注意的问题

1、如果对状态机工作流发送的事件不是工作流当前的状态中所需要的,则会抛错。

2、因为Web不能保持程序当前的很多状态信息,因此需要自己处理对工作流引擎、实例等的信息保存。

3、在工作流引擎启动之前加载需要的服务和绑定需要工作流处理的事件。

4、同一工作流加载同一服务多次会出错。

5、工作流引擎可以不需要启动动作,而是用创建实例的启动方法来同时启动工作流引擎,但在持久化恢复时需要启动工作流引擎。



因为纯属个人练习性质的项目,该项目代码比较凌乱,且处理逻辑不够清晰,较复杂,而且是第一次写Blog,请大家谅解。谢谢!

使用本例需要改变代码中连接SQL Server数据库的字符串。

本例下载