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

2.3.2.2 使用WorkflowCompiler类

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

 《WF编程》系列之10 - 编译工作流:使用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.

你可能感兴趣的:(workflow)