c# - TPL 队列处理

标签 c# queue task-parallel-library

我目前正在做一个项目,我需要排队处理一些作业,这是要求:

  1. 必须一次处理一个作业
  2. 排队的项目必须能够等待

所以我想要类似于:

Task<result> QueueJob(params here)
{
   /// Queue the job and somehow return a waitable task that will wait until the queued job has been executed and return the result.
}

我试过让后台运行的任务只从队列中取出项目并处理作业,但困难在于从后台任务到方法。

如果需要,我可以采用仅在 QueueJob 方法中请求完成回调的方式,但如果我可以返回一个透明的 Task 以允许您等待要处理的作业(即使队列中有它前面的作业)。

最佳答案

您可能会找到 TaskCompletionSource<T> 很有用,它可以用来创建 Task恰好在您想要的时候完成。如果将它与 BlockingCollection<T> 结合使用,你会得到你的队列:

class JobProcessor<TInput, TOutput> : IDisposable
{
    private readonly Func<TInput, TOutput> m_transform;

    // or a custom type instead of Tuple
    private readonly
        BlockingCollection<Tuple<TInput, TaskCompletionSource<TOutput>>>
        m_queue =
        new BlockingCollection<Tuple<TInput, TaskCompletionSource<TOutput>>>();

    public JobProcessor(Func<TInput, TOutput> transform)
    {
        m_transform = transform;
        Task.Factory.StartNew(ProcessQueue, TaskCreationOptions.LongRunning);
    }

    private void ProcessQueue()
    {
        Tuple<TInput, TaskCompletionSource<TOutput>> tuple;
        while (m_queue.TryTake(out tuple, Timeout.Infinite))
        {
            var input = tuple.Item1;
            var tcs = tuple.Item2;

            try
            {
                tcs.SetResult(m_transform(input));
            }
            catch (Exception ex)
            {
                tcs.SetException(ex);
            }
        }
    }

    public Task<TOutput> QueueJob(TInput input)
    {
        var tcs = new TaskCompletionSource<TOutput>();
        m_queue.Add(Tuple.Create(input, tcs));
        return tcs.Task;
    }

    public void Dispose()
    {
        m_queue.CompleteAdding();
    }
}

关于c# - TPL 队列处理,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14929592/

相关文章:

c# - MongoDB C# 驱动程序 - 更新嵌入式文档数组中的所有字段

c# - PayPal 集成 (IPN) - 真的有可能吗?

c# - WPF Tab 键导航

java - 在方法中使用实例变量 Linkedlist() - 为什么我必须在方法体中创建 new

java - 有没有在数组之上实现队列并自动调整大小的java库?

c# - 使用 AutoMapper 将嵌套元素映射到相关列表

c++ - 在 C/C++ 中实现工作窃取队列?

.net - 在哪里可以找到 4.0 的 TPL 数据流版本?

c# - 在哪里为基于任务的异步方法定义回调

c# - 并行代码可扩展性差