《WF编程》系列之10 - 编译工作流:使用WorkflowCompiler类
2 min read

《WF编程》系列之10 - 编译工作流:使用WorkflowCompiler类

2.3.2.2 使用WorkflowCompiler类

我们可以使用WorkflowCompiler类和WorkflowCompilerResults类以编程的方式来编译工作流定义并得到一个程序集.其实Wfc.exe就是通过使用WorkflowCompiler类来执行编译过程的.下图是和编译相关的几个类:

 

在使用WorkflowCompiler类之前,我们需要设置WorkflowCompilerParameters对象.我们可以使用这个参数对象来引用任何包含自定义活动的程序集.

WorkflowCompiler compiler = new WorkflowCompiler();
WorkflowCompilerParameters parameters;
parameters = new WorkflowCompilerParameters();
parameters.GenerateInMemory = true;
parameters.ReferencedAssemblies.Add("chapter2_Host.exe");
string[] xomlFiles = { @"..\..\purexaml\purexaml3.xoml" };
WorkflowCompilerResults compilerResults;
compilerResults = compiler.Compile(parameters, xomlFiles);

注意,Compile方法的参数之一是字符串数组,我们可以通过它一次性传递多个XOML文件给WorkflowCompiler.如果编译失败,编译结果的Errors属性将包含错误的详细信息.

if (compilerResults.Errors.Count > 0)
{
foreach (CompilerError error in compilerResults.Errors)
{
Console.WriteLine(error.ErrorText);
}
}

使用WorkflowCompiler编译的工作流和使用Wfc.exe编译的工作流程序集差别不大.我们不再需要Type.GetType方法,而是直接从编译结果中获取类型.

using (WorkflowRuntime runtime = new WorkflowRuntime())
{
using (AutoResetEvent waitHandle = new AutoResetEvent(false))
{
runtime.WorkflowCompleted += delegate { waitHandle.Set(); };
runtime.WorkflowTerminated += delegate { waitHandle.Set(); };
Type workflowType;
workflowType = compilerResults.CompiledAssembly.GetType("MyWorkflow");
WorkflowInstance instance
= runtime.CreateWorkflow(workflowType);
instance.Start();
waitHandle.WaitOne();
}
}

关于编译过程,还有两个有趣的实现细节值得在这里提一下:

  • WorkflowCompiler每次调用Compile方法时都创建新的应用程序域(AppDomain),所以在设计应用程序时,要提防应用程序在它的生命周期内多次调用Compile方法.
  • 如果GenerateInMemory参数设置为true而且可以编译成功,Compile方法会自动加载新的程序集到当前应用程序域(AppDomain),.如果你不需要立即加载程序集,请将GenerateInMemory设置为false.