我创建了一个执行Web请求并将结果存储到数据库中的活动。我发现对于这些长时间运行的活动,我应该编写一些不同的代码,以便不会阻止工作流引擎线程。
public sealed class WebSaveActivity : NativeActivity
{
protected override void Execute(NativeActivityContext context)
{
GetAndSave(); // This takes 1 hour to accomplish.
}
}
我应该如何重写此活动以满足长时间运行活动的要求
答案 0 :(得分:1)
你可以使用例如在现有流程中生成一个线程。 ThreadPool.QueueUserWorkItem()因此,如果需要,您的工作流程的其余部分将继续运行。但是,请务必先了解多线程和线程同步的含义。 或者您可以查看Hangfire或类似组件,将整个作业卸载到不同的进程中。
编辑:
根据您的评论,您可以查看基于任务的异步模式(TAP):Link 1,Link 2这将为您提供一个很好的编写代码模型,继续处理可能的事情在等待长时间运行的结果直到它返回时完成。但是,我不确定这是否涵盖了您的所有需求。特别是在Windows Workflow Foundation中,您可能希望查看某种形式的workflow hibernation/persistence。
答案 1 :(得分:0)
这种情况是使用WF持久性功能的地方。它允许您将工作流实例持久保存到数据库,以允许完成一些长时间运行的操作。完成后,第二个线程或进程可以重新保存工作流实例并允许其恢复。
首先,您向工作流应用程序指定工作流实例商店。 Microsoft提供了您可以使用的SQL workflow instance store implementation,并提供了可以在SQL Server上运行的SQL脚本。
namespace MySolution.MyWorkflowApp
{
using System.Activities;
using System.Activities.DurableInstancing;
using System.Activities.Statements;
using System.Threading;
internal static class Program
{
internal static void Main(string[] args)
{
var autoResetEvent = new AutoResetEvent(false);
var workflowApp = new WorkflowApplication(new Sequence());
workflowApp.InstanceStore = new SqlWorkflowInstanceStore("server=mySqlServer;initial catalog=myWfDb;...");
workflowApp.Completed += e => autoResetEvent.Set();
workflowApp.Unloaded += e => autoResetEvent.Set();
workflowApp.Aborted += e => autoResetEvent.Set();
workflowApp.Run();
autoResetEvent.WaitOne();
}
}
}
您的活动将启动将实际执行保存操作的辅助进程/线程。有多种方法可以做到这一点:
您的活动如下:
public sealed class WebSaveActivity : NativeActivity
{
public InArgument<MyBigObject> ObjectToSave { get; set; }
protected override bool CanInduceIdle
{
get
{
// This notifies the WF engine that the activity can be unloaded / persisted to an instance store.
return true;
}
}
protected override void Execute(NativeActivityContext context)
{
var currentBigObject = this.ObjectToSave.Get(context);
currentBigObject.WorkflowInstanceId = context.WorkflowInstanceId;
StartSaveOperationAsync(this.ObjectToSave.Get(context)); // This method should offload the actual save process to a thread or even a web method, then return immediately.
// This tells the WF engine that the workflow instance can be suspended and persisted to the instance store.
context.CreateBookmark("MySaveOperation", AfterSaveCompletesCallback);
}
private void AfterSaveCompletesCallback(NativeActivityContext context, Bookmark bookmark, object value)
{
// Do more things after the save completes.
var saved = (bool) value;
if (saved)
{
// yay!
}
else
{
// boo!!!
}
}
}
书签创建向WF引擎发出信号,表明工作流实例可以从内存中卸载,直到唤醒工作流实例。
在您的方案中,您希望在长时间保存操作完成后恢复工作流程。让我们假设StartSaveOperationAsync
方法将一条小消息写入某种队列,第二个线程或进程轮询执行保存操作:
public static void StartSaveOperationAsync(MyBigObject myObjectToSave)
{
var targetQueue = new MessageQueue(".\private$\pendingSaveOperations");
var message = new Message(myObjectToSave);
targetQueue.Send(message);
}
在我的第二个过程中,我可以轮询队列以获取新的保存请求并重新保存持久的工作流实例,以便在保存操作完成后恢复。假设以下方法位于不同的控制台应用程序中:
internal static void PollQueue()
{
var targetQueue = new MessageQueue(@".\private$\pendingSaveOperations");
while (true)
{
// This waits for a message to arrive on the queue.
var message = targetQueue.Receive();
var myObjectToSave = message.Body as MyBigObject;
// Perform the long running save operation
LongRunningSave(myObjectToSave);
// Once the save operation finishes, you can resume the associated workflow.
var autoResetEvent = new AutoResetEvent(false);
var workflowApp = new WorkflowApplication(new Sequence());
workflowApp.InstanceStore = new SqlWorkflowInstanceStore("server=mySqlServer;initial catalog=myWfDb;...");
workflowApp.Completed += e => autoResetEvent.Set();
workflowApp.Unloaded += e => autoResetEvent.Set();
workflowApp.Aborted += e => autoResetEvent.Set();
// I'm assuming the object to save has a field somewhere that refers the workflow instance that's running it.
workflowApp.Load(myObjectToSave.WorkflowInstanceId);
workflowApp.ResumeBookmark("LongSaveOperation", true); // The 'true' parameter is just our way of saying the save completed successfully. You can use any object type you desire here.
autoResetEvent.WaitOne();
}
}
private static void LongRunningSave(object myObjectToSave)
{
throw new NotImplementedException();
}
public class MyBigObject
{
public Guid WorkflowInstanceId { get; set; } = Guid.NewGuid();
}
现在,长时间运行的保存操作不会妨碍工作流引擎,并且通过不将工作流实例长时间保留在内存中,可以更有效地利用系统资源。