通过XOML或者XAML的方式创建和启动工作流
在Workflow Foundation中,提供了多种设计工作流的方式。例如
- 纯代码的方式(C#)
- 代码分离的方式(XOML+C#)
以上两种,可以从Visual Studio提供的项目模板中选择
他们在设计器中看起来是如下的
xoml是一种特殊的XML格式,它可以描述流程。例如下面这样的一个流程
它用XOML描述的话,大致如下
<SequentialWorkflowActivity x:Class="WorkflowConsoleApplication1.Workflow2" x:Name="Workflow2" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" xmlns="http://schemas.microsoft.com/winfx/2006/xaml/workflow">
<CodeActivity x:Name="codeActivity1" />
<IfElseActivity x:Name="ifElseActivity1">
<IfElseBranchActivity x:Name="ifElseBranchActivity1">
<CodeActivity x:Name="codeActivity2" />
</IfElseBranchActivity>
<IfElseBranchActivity x:Name="ifElseBranchActivity2">
<CodeActivity x:Name="codeActivity3" />
</IfElseBranchActivity>
</IfElseActivity>
</SequentialWorkflowActivity>
虽然上面这样的XML确实可以描述流程,但流程所涉及的一些特殊逻辑,则可能还是需要写代码。这两个部分,最终还是需要编译成一个类型出来。
所以,实际上上面的两种方式没有本质的区别。
这里要介绍的是一种纯XOML的方式:我们能不能全部用XOML来实现对流程的定义?如果这样的话,我们就可以实现更加动态化的流程,因为如果存在额外的代码,就无需编译,而是可以通过直接修改XOML文件就完成流程的修改。
答案是:可以的。但有一个前提,既然我们不想用代码,那么流程中所使用的Activity就应该都是可以通过配置即可完成工作的。
通常,我们需要编写自定义Activity来实现这样的需求。
例如,我们可以做一个最简单的Activity,它只是根据用户输入的一个参数,负责在屏幕上打印一个消息。
using System;
using System.ComponentModel;
using System.ComponentModel.Design;
using System.Collections;
using System.Linq;
using System.Workflow.ComponentModel;
using System.Workflow.ComponentModel.Design;
using System.Workflow.ComponentModel.Compiler;
using System.Workflow.ComponentModel.Serialization;
using System.Workflow.Runtime;
using System.Workflow.Activities;
using System.Workflow.Activities.Rules;
namespace WorkflowConsoleApplication1
{
public partial class MyActivity :Activity
{
public MyActivity()
{
InitializeComponent();
}
public string Message { get; set; }
protected override ActivityExecutionStatus Execute(ActivityExecutionContext executionContext)
{
Console.WriteLine(Message);
return base.Execute(executionContext);
}
}
}
为了演示如何直接通过XOML定义的方式使用这个自定义Activity,我们可以通过下面的代码生成一段XOML
var workflow = new SequenceActivity();
workflow.Activities.Add(new MyActivity() { Message = "Hello,World" });
var serializer = new WorkflowMarkupSerializer();
serializer.Serialize(
XmlWriter.Create("test.xoml",new XmlWriterSettings(){CloseOutput=true}), workflow);
以上代码的意思是,创建一个SequenceActivity,并且在其中添加我们自定义的这个Activity。
代码执行完成之后,生成的那个test.xoml的内容如下
<?xml version="1.0" encoding="utf-8"?><SequenceActivity x:Name="SequenceActivity" xmlns:ns0="clr-namespace:WorkflowConsoleApplication1;Assembly=WorkflowConsoleApplication1, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" xmlns="http://schemas.microsoft.com/winfx/2006/xaml/workflow">
<ns0:MyActivity Message="Hello,World" x:Name="myActivity1" />
</SequenceActivity>
这里请仔细观察一下与之前那段xoml的区别
- 顶层的Activity,不再包含x:Class定义。(因为现在没有了代码文件,Class就无从说起了)
- 内部所有的自定义Activity,都需要预先定义有关的namespace,引用到它们所在的Assembly
- 自定义Activity的设置,直接通过属性完成
那么,如何利用上面这样的定义,去创建并运行流程呢?
var instance2 = workflowRuntime.CreateWorkflow(
XmlReader.Create("test.xoml"));
instance2.Start();
如果我们需要修改流程定义,则直接可以修改那个XOML文件,而无需编译程序。例如我们将Message修改为“Hello,Workflow”
则运行结果如下
其实,只要我们愿意,我们甚至可以将这些定义的字符串保存在爱数据库中,然后通过下面代码创建并启动流程
var definition ="<?xml version=\"1.0\" encoding=\"utf-8\"?><SequenceActivity x:Name=\"SequenceActivity\" xmlns:ns0=\"clr-namespace:WorkflowConsoleApplication1;Assembly=WorkflowConsoleApplication1, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null\" xmlns:x=\"http://schemas.microsoft.com/winfx/2006/xaml\" xmlns=\"http://schemas.microsoft.com/winfx/2006/xaml/workflow\"><ns0:MyActivity Message=\"Hello,Workflow\" x:Name=\"myActivity1\" /></SequenceActivity>";
//这里还可以从数据库将该定义读取出来
var reader = XmlReader.Create(new StringReader(definition));
var workflow = workflowRuntime.CreateWorkflow(reader);
workflow.Start();