并行的 foreach 循环 - 奇怪的行为

10
下面的代码简单地创建了一个随机数列表,并在并行foreach循环中计算每个列表的累积和。为什么我收到的评估次数少于“numLists”?通常约为9990。我猜这与线程安全有关。有什么替代方法吗?(我是C#的初学者,所以希望我使用正确的术语)谢谢。
using System;  
using System.Collections.Generic;  
using System.Threading.Tasks;  

namespace testParallelForeach
{
    class Program
    {
        static void Main(string[] args)
        {

            List<List<double>> bsData = new List<List<double>>();
            List<List<double>> cumsumDataP = new List<List<double>>();
            int numLists = 10000;
            int myLen = 400;
            Random rand = new Random();
            for (int i = 0; i < numLists; i++)
            {
                bsData.Add(new List<double>());
                for (int j = 0; j < myLen; j++)
                {
                    bsData[i].Add(rand.NextDouble());
                }
            }
            Parallel.ForEach(bsData, a => cumsumDataP.Add(CumulativeSumParallel(a)));
            Console.WriteLine("cumsumDataP.Count={0}", cumsumDataP.Count);
            Console.ReadKey();

        }

        public static List<double> CumulativeSumParallel(List<double> singleRetSeries)
        {
            int r = singleRetSeries.Count;
            List<double> cumsumList = new List<double>();

            cumsumList.Add(singleRetSeries[0]);
            for (int i = 1; i < r; i++)
            {
                cumsumList.Add(cumsumList[i - 1] + singleRetSeries[i]);
            }
            return cumsumList;
        }
    }
}
1个回答

15

List<T> 的确不是线程安全的,因此 cumsupDataP.Add(...) 会以不可预测的方式丢失数据。

请将该行替换为:

ConcurrentBag<List<double>> cumsumDataP = new ConcurrentBag<List<double>>();

简单来说,只要使用 ConcurrentBag<T> 就可以实现并发安全的添加和删除元素,尽管它是 无序 的,但由于线程无法预测顺序,所以这是可以接受的。 ;p


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接