我是TPL Dataflow
的新手。我有一个项目编号列表,我需要处理。一个项目可能有大约8000
个项目,我需要获取项目中每个项目的数据,然后将这些数据推送到5个独立的服务器中。
这是我迄今编码的内容。我陷入了如何将这些数据加载到5台服务器的步骤。我不确定这是否编码正确。任何建议都非常感谢。
public static bool PushData(string projectId)
{
var linkCompletion = new DataflowLinkOptions
{
PropagateCompletion = true
};
var projectItems = new TransformBlock<ProjectDTO, ProjectDTO>(
dto => dto.GetItemData(dto), new ExecutionDataflowBlockOptions { MaxDegreeOfParallelism = 5 });
var itemData = new ActionBlock<ProjectDTO>(
dto => PostEachServerAsync(dto, "server1", "setmemcache"), new ExecutionDataflowBlockOptions { MaxDegreeOfParallelism = 5 });
projectItems.LinkTo(projectRules, linkCompletion);
IList<ProjectDTO> dtoList = new List<ProjectDTO>();
dtoList = MemcachedDTO.GetDataByProject(projectId);
foreach (ProjectDTOd in dtoList)
{
projectItems.Post(d);
}
projectItems.Complete();
projectItems.Completion.Wait();
return false;
}
现在这是我的代码 - 但它没有正确完成 - 有谁能告诉我我做错了什么?
[HttpGet]
public HttpResponseMessage ReloadItem(string projectQuery)
{
try
{
var linkCompletion = new DataflowLinkOptions
{
PropagateCompletion = true
};
IList<string> projectIds = projectQuery.Split(',').ToList();
IEnumerable<string> serverList = ConfigurationManager.AppSettings["ServerList"].Split(',').Cast<string>();
var iR = new TransformBlock<MemcachedDTO, MemcachedDTO>(
dto => dto.GetIR(dto), new ExecutionDataflowBlockOptions { MaxDegreeOfParallelism = 3 });
var pR = serverList.Select(
i => new { Id = i, Action = new ActionBlock<MemcachedDTO>(dto => PostEachServerAsync(dto, i, "set"), new ExecutionDataflowBlockOptions { MaxDegreeOfParallelism = 3 }) });
List<MemcachedDTO> dtoList = new List<MemcachedDTO>();
foreach (string pid in projectIds)
{
IList<MemcachedDTO> dtoTemp = new List<MemcachedDTO>();
dtoTemp = MemcachedDTO.GetItemIdsByProject(pid);
dtoList.AddRange(dtoTemp);
}
foreach (var action in pR)
{
iR.LinkTo(action.Action, linkCompletion);
}
foreach (MemcachedDTO d in dtoList)
{
iR.Post(d);
}
iR.Complete();
foreach (var action in pR)
{
action.Action.Completion.Wait();
}
return Request.CreateResponse(HttpStatusCode.OK, new { message = projectIds.ToString() + " reload success" });
}
catch (Exception ex)
{
return Request.CreateResponse(HttpStatusCode.InternalServerError, new { message = ex.Message.ToString() });
}
}
答案 0 :(得分:1)
您的代码根本没有编译,您如何运行它?
首先,请勿使用.Wait()
阻止您的主题,请在此处使用async/await
pattern。其次,您需要BroadcastBlock
来通知您的数据超过1个块。第三,你需要5个不同的ActionBlock
s,而不是1个并行度5.第四,你等待错误的Completion
任务 - 等待最后一个块完成,而不是第一个,所以在您的情况下,您需要等待WhenAll
method的5个块完成。
所以你的代码可能是这样的(我假设projectRules
和itemsData
是同一个块):
public static async Task<bool> PushData(string projectId)
{
var linkCompletion = new DataflowLinkOptions
{
PropagateCompletion = true
};
var projectItems = new TransformBlock<ProjectDTO, ProjectDTO>(
dto => dto.GetItemData(dto), new ExecutionDataflowBlockOptions { MaxDegreeOfParallelism = 5 });
var broadcast = new BroadcastBlock<ProjectDTO>();
projectItems.LinkTo(broadcast, linkCompletion);
var pR = serverList.Select(
i => new { Id = i, Action = new ActionBlock<MemcachedDTO>(dto => PostEachServerAsync(dto, i, "set"), new ExecutionDataflowBlockOptions { MaxDegreeOfParallelism = 3 }) });
foreach (var action in pR)
{
broadcast.LinkTo(action.Action, linkCompletion);
}
var dtoList = MemcachedDTO.GetDataByProject(projectId);
foreach (var d in dtoList)
{
projectItems.Post(d);
}
projectItems.Complete();
// wait all the action blocks to finish
await Task.WhenAll(projectRules1.Completion, projectRules2.Completion, projectRules3.Completion, projectRules4.Completion, projectRules5.Completion);
return false;
}