如何在C#中为多个值运行单个进程?

时间:2019-04-11 09:33:40

标签: c# multithreading linq parallel-processing task

我需要为符号列表运行单个进程。因此,我有一个符号列表,如AAPL,FB,QQQ,MSFT,IBM,而且我还有一个函数,其中包含给定符号的计算逻辑。因此,请执行我使用Task的逻辑。

List<string> symbolList = new List<string>() {"AAPL","QQQ","FB","MSFT","IBM"};
Task[] taskArray = new Task[symbolList.Count];
for(int i=0; i<taskArray.Length; i++)
{
   taskArray[i] = Task.Factory.StartNew(() =>
       {
          criteriaEvalution.Evaluate(finalArray, false, new List<parseObj>(), ref builder, symbolList.IndexOf(i));
       });
}
Task.WaitAll(taskArray);

因此,当我运行此代码时,它将向我显示

之类的错误
  

索引超出范围。必须为非负数,并且小于集合的大小。

错误显示在行上:

criteriaEvalution.Evaluate(finalArray, false, new List<parseObj>(), ref builder, symbolList.IndexOf(i));

我要将符号名称传递给函数的地方。

所以请问我能解决这个问题吗? 我已经在google上检查过,这是因为索引超出范围。我已经通过设置调试器进行了检查,但是我不知道如何检查任务区域。

4 个答案:

答案 0 :(得分:1)

Parallel.Foreach可以做到这一点。这里有一个示例:

public static void DoIt(string a)
{
    Console.WriteLine(a);
}

public static void Main(string[] args)
{
    List<string> symbolList = new List<string>() { "AAPL", "QQQ", "FB", "MSFT", "IBM" };
    Parallel.ForEach(symbolList, a => DoIt(a));
}

您可以使用ParallelOptions控制呼叫:

ParallelOptions options = new ParallelOptions() { MaxDegreeOfParallelism = 10 };
Parallel.ForEach(symbolList, options, a => DoIt(a));

答案 1 :(得分:1)

首先,在这种情况下(如果您写入集合),我将使用并发集合ConcurrentBag。当多个线程将要访问实例时,并发集合将锁定该实例。另外,我建议将Task.WhenAllawait关键字一起使用。 Task.WaitAll会造成死锁。因此,您可以尝试以下示例:

public static async Task Main()
{
    var symbolList = new ConcurrentBag<string> { "AAPL", "QQQ", "FB", "MSFT", "IBM" };
    var taskArray = new List<Task>();

    foreach (var s in symbolList)
    {
        var task = Task.Run(() =>
        {
            Process(s);
        });

        taskArray.Add(task);
    }

    await Task.WhenAll(taskArray);
}

答案 2 :(得分:0)

可能您应该替换:

symbolList.IndexOf(i)

...具有:

symbolList[i]

答案 3 :(得分:0)

除了您有非法代码(symbolList.IndexOf(i)应该是symbolList[i])之外,您面临的问题是在调用lambda之前没有捕获循环变量

这是简单的解决方法:

List<string> symbolList = new List<string>() { "AAPL", "QQQ", "FB", "MSFT", "IBM" };
Task[] taskArray = new Task[symbolList.Count];
for (int i = 0; i < taskArray.Length; i++)
{
    string symbol = symbolList[i];
    taskArray[i] = Task.Factory.StartNew(() =>
    {
        criteriaEvalution.Evaluate(finalArray, false, new List<parseObj>(), ref builder, symbol);
    });
}
Task.WaitAll(taskArray);

在开始任务之前注意第string symbol = symbolList[i];行。

如果不执行此操作,循环将在任务开始之前完成,然后i等于taskArray.Length,因此会出现“索引超出范围”错误。

我建议的另一个建议是,您应该使用Microsoft的Reactive Framework(又名Rx)-NuGet System.Reactive并添加using System.Reactive.Linq;-然后您可以执行以下操作:

List<string> symbolList = new List<string>() { "AAPL", "QQQ", "FB", "MSFT", "IBM" };

var query =
    from symbol in symbolList.ToObservable()
    from e in Observable.Start(() =>
        criteriaEvalution.Evaluate(finalArray, false, new List<parseObj>(), ref builder, symbol))
    select e;

query.ToArray().Wait();

比任务简单,干净得多。然后,您也可以对结果使用LINQ运算符。