C# List 並發丟數據問題原因及解決方案

項目中出瞭個 BUG,就在我眼皮子底下,很明顯的一個 BUG,愣是看瞭兩天才看出來。

我有多個任務並發,任務執行完成後都有一個返回結果,我用一個 List 將結果收集起來,等所有任務完成後,發送出去。結果一直 丟數據。

我反復檢查邏輯都沒有問題,最後恍然 List 是非線程安全的。

大傢都知道 List 是非線程安全的,但是如果僅有 Add 操作呢?估計有些人就會認為沒問題。

下面的代碼,期望輸出的結果是 1000,然而,註釋掉 lock 後,結果就不一樣瞭。

class Program
{
 static List<Person> persons;

 static void Main(string[] args)
 {
  persons = new List<Person>();

  object sync = new object();

  Parallel.For(0, 1000, (i) =>
  {
   Person person = new Person
   {
    ID = i,
    Name = "name" + i
   };
   lock (sync)
    persons.Add(person);
  });

  Console.WriteLine(persons.Count);
  Console.ReadLine();
 }

 class Person
 {
  public int ID { get; set; }
  public string Name { get; set; }
 }
}

利用安全集合ConcurrentBag取代list

測試程序

using System;
using System.Collections.Concurrent;
using System.Collections.Generic;
using System.Diagnostics;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
 
namespace MyConcurrent
{
  class Program
  {
    /// <summary>
    /// ConcurrentBag並發安全集合
    /// </summary>
    public static void ConcurrentBagWithPallel()
    {
      ConcurrentBag<int> list = new ConcurrentBag<int>();
      Parallel.For(0, 10000, item =>
      {
        list.Add(item);
      });
      Console.WriteLine("ConcurrentBag's count is {0}", list.Count());
      int n = 0;
      foreach (int i in list)
      {
        if (n > 10)
          break;
        n++;
        Console.WriteLine("Item[{0}] = {1}", n, i);
      }
      Console.WriteLine("ConcurrentBag's max item is {0}", list.Max());
    }
    
 
    /// <summary>
    /// 函數入口
    /// </summary>
    /// <param name="args"></param>
    static void Main(string[] args)
    {
      Console.WriteLine("ConcurrentBagWithPallel is runing" );
      ConcurrentBagWithPallel();
 
      Console.Read();
    }

以上就是C# List 並發丟數據問題原因及解決方案的詳細內容,更多關於C# List 並發丟數據的資料請關註WalkonNet其它相關文章!

推薦閱讀: