使用并行库添加到列表时保证线程安全的正确方法

时间:2012-01-09 23:10:57

标签: c# multithreading thread-safety parallel-extensions

我遍历一个连接字符串数组,并在每个循环中提取一些信息并添加到列表中。现在,我想使用并行库使其成为多线程,但我不确定该库是否保证对列表的写入是线程安全的,还是我需要使用锁定:

List<SomeType> list = new List<SomeType>();

settings.AsParallel().ForAll(setting => 
{
    list.AddRange(GetSomeArrayofSomeType(setting)); /// DO I NEED TO DO LOCKING HERE???
})

1 个答案:

答案 0 :(得分:10)

写入列表对于多线程写入确实不安全。您需要使用lock来同步访问权限,或者使用专为多线程访问设计的ConcurrentQueue集合。

锁示例(假设list是方法的局部)

List<SomeType> list = new List<SomeType>();
settings.AsParallel().ForAll(setting => { 
  lock (list) {
    list.AddRange(GetSomeArrayofSomeType(setting)); 
  }
});

或者更好的是使用SelectMany代替ForEach

var list = settings
  .AsParallel()
  .SelectMany(setting => GetSomeArrayOfSomeType(setting))
  .ToList();