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.