我正在使用Microsoft Orleans作为基础构建一个工作流引擎,因为它提供了许多有用的功能,例如自动分配工作和处理故障转移。我有三种类型的grains:
- 工作流:保存工作流程信息以及应该执行工作块的顺序。 - 工作块:实际执行工作的部分。 - 执行:工作流程的单个执行。
我的问题是,当运行大量的当前执行(即> 1000)时,性能真的很差。我已经进行了一些分析,并将其缩小到grains之间发生的通信。是否有任何进一步改进的方法?以下是我的代码概要及grains相互作用:
执行grain在循环中等待下一个工作块并调用工作块的execute。正是这种grains之间的不断调用导致了其中一个测试工作流的执行时间从单次执行10秒增加到超过1000次执行时约5分钟。这可以改善吗,或者应该重新设计解决方案以消除grain通信?
- 工作流:保存工作流程信息以及应该执行工作块的顺序。 - 工作块:实际执行工作的部分。 - 执行:工作流程的单个执行。
我的问题是,当运行大量的当前执行(即> 1000)时,性能真的很差。我已经进行了一些分析,并将其缩小到grains之间发生的通信。是否有任何进一步改进的方法?以下是我的代码概要及grains相互作用:
执行grain在循环中等待下一个工作块并调用工作块的execute。正是这种grains之间的不断调用导致了其中一个测试工作流的执行时间从单次执行10秒增加到超过1000次执行时约5分钟。这可以改善吗,或者应该重新设计解决方案以消除grain通信?
[StorageProvider(ProviderName = "WorkflowStore")]
[Reentrant]
[StatelessWorker]
public class Workflow : Grain<WorkflowState>, IWorkflow
{
public Task<BlockRef> GetNext(Guid currentBlockId, string connectionName)
{
//Lookup the next work block
}
}
[Reentrant]
[StatelessWorker]
public class WorkBlock : Grain<WorkBlock State>, IWorkBlock
{
public Task<string> Execute(IExecution execution)
{
//Do some work
}
}
[StorageProvider(ProviderName = "ExecutionStore")]
public class Execution : Grain<ExecutionState>, IExecution, IRemindable
{
private async Task ExecuteNext(bool skipBreakpointCheck = false)
{
if (State.NextBlock == null)
{
await FindAndSetNext(null, null);
}
...
var outputConnection = await workblock.Execute();
if (!string.IsNullOrEmpty(outputConnection))
{
await FindAndSetNext(State.NextBlock.Id, outputConnection);
ExecuteNext().Ignore();
}
}
private async Task FindAndSetNext(Guid? currentId, string outputConnection)
{
var next = currentId.HasValue ? await _flow.GetNextBlock(currentId.Value, outputConnection) : await _flow.GetNextBlock();
...
}
}