欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C#并發(fā)實戰(zhàn)記錄之Parallel.ForEach使用

 更新時間:2019年08月11日 10:10:47   作者:<漁人>  
這篇文章主要給大家介紹了關于C#并發(fā)實戰(zhàn)記錄之Parallel.ForEach使用的相關資料,文中通過示例代碼介紹的非常詳細,對大家學習或者使用C#具有一定的參考學習價值,需要的朋友們下面來一起學習學習吧

前言:

最近給客戶開發(fā)一個伙食費計算系統(tǒng),大概需要計算2000個人的伙食。需求是按照員工的預定報餐計劃對消費記錄進行檢查,如有未報餐有刷卡或者有報餐沒刷卡的要進行一定的金額扣減等一系列規(guī)則。一開始我的想法比較簡單,直接用一個for循環(huán)搞定,統(tǒng)計結果倒是沒問題,但是計算出來太慢了需要7,8分鐘。這樣系統(tǒng)服務是報超時錯誤的,讓人覺得有點不太爽。由于時間也不多就就先提交給用戶使用了,后面邏輯又增加了,計算時間變長,整個計算一遍居然要將近10分鐘了。這個對用戶來說是能接收的(原來自己手算需要好幾天呢),但是我自己接受不了,于是就開始優(yōu)化了,怎么優(yōu)化呢,用多線程唄。

一提到多線程,最先想到的是Task了,畢竟.net4.0以上Task封裝了很多好用的方法。但是Task畢竟是多開一些線程去執(zhí)行任務,最后整合結果,這樣可以快一些,但我想更加快速一些,于是想到了另外一個對象:Parallel。之前在維護代碼是確實有遇到過別人寫的Parallel.Invoke,只是指定這個函數的作用是并發(fā)執(zhí)行多項任務,如果遇到多個耗時的操作,他們之間又不貢獻變量這個方法不錯。我的情況是要并發(fā)執(zhí)行一個集合,于是就用了List.ForAll 這個方法其實是拓展方法,完整的調用為:List.AsParallel().ForAll,需要先轉換成支持并發(fā)的集合,等同于Parallel.ForEach,目的是對集合里面的元素并發(fā)執(zhí)行一系列操作。

于是乎,把原來的foreach換成了List.AsParallel().ForAll,運行起來,果然速度驚人,不到兩分鐘就插入結果了,但最后卻是報主鍵重復的錯誤,這個錯誤的原因是,由于使用了并發(fā),這個時候變量自增,其實是在強著自增,當多個線程同時獲取到了id值,都去自增然后就重復了,舉個例子如下:

int num = 1;
      List<int> list = new List<int>();
      for (int i = 1; i <= 2000; i++)
      {
        list.Add(i);
      }
      Console.WriteLine($"num初始值為:" + num.ToString());
      list.AsParallel().ForAll(n =>
      {
        num++;
      });
      Console.WriteLine($"不加鎖,并發(fā){list.Count}次后為:" + num.ToString());
      Console.ReadKey();

這段代碼是讓一個變量執(zhí)行2000次自增,正常結果應該是2001,但實際結果如下:

有經驗的同學,立馬能想到需要加鎖了,C#內置了很多鎖對象,如lock 互斥鎖,Interlocked 內部鎖,Monitor 這幾個比較常見,lock內部實現(xiàn)其實就是使用了Monitor對象。對變量自增,Interlocked對象提供了,變量自增,自減、或者相加等方法,我們使用自增方法Interlocked.Increment,函數定義為:int Increment(ref int num),該對象提供原子性的變量自增操作,傳入目標數值,返回或者ref num都是自增后的結果。 在之前的基礎上我們增加一些代碼:

num = 1;
      Console.WriteLine($"num初始值為:" + num.ToString());
      list.AsParallel().ForAll(n =>
      {
        Interlocked.Increment(ref num);
      });
      Console.WriteLine($"使用內部鎖,并發(fā){list.Count}次后為:" + num.ToString());
      Console.ReadKey();

我們來看運行結果:

加了鎖之后ID重復算是解決了,其實別高興太早,由于正常的環(huán)境有了ID我們還有用這些ID來構建對象呢,于是又寫了寫代碼,用集合來添加這些ID,為了更真實的模擬生產環(huán)境,我在forAll里面又加了一層循環(huán)代碼如下:

num = 1;
      Random random = new Random();
      var total = 0;
      var m = new ConcurrentBag<int>();
      list.AsParallel().ForAll(n =>
      {
        var c = random.Next(1, 50);
        Interlocked.Add(ref total, c);
        for (int i = 0; i < c; i++)
        {
          Interlocked.Increment(ref num);
          m.Add(num);
        }
      });
      Console.WriteLine($"使用內部鎖,并發(fā)+內部循環(huán){list.Count}次后為:" + num.ToString());
      Console.WriteLine($"實際值為:{total + 1}");
      var l = m.GroupBy(n => n).Where(o => o.Count() > 1);
      Console.WriteLine($"并發(fā)里面使用安全集合ConcurrentBag添加num,集合重復值:{l.Count()}個");
      Console.ReadKey();

上面的代碼里面我用到了線程安全集合ConcurrentBag<T>它的命名空間是:using System.Collections.Concurrent,盡管使用了線程安全集合,但是在并發(fā)面前仍然是不安全的,到了這里其實比較郁悶了,自增加鎖,安全集合內部應該也使用了鎖,但還是重復了。有點說不過去了,想想多線程執(zhí)行時有個上下文對象,即當多個線程同時執(zhí)行任務,共享了變量他們一開始傳進去的對象數值應該是相同的,由于變量自增時加了鎖,所以ID是不會重復了。我猜測問題應該出在Add方法了,就是說當num值自增后還沒有來得及傳出去就已經執(zhí)行了Add方法,故添加了重復變量。于是乎,我重新寫了段代碼,讓ID自增和集合添加都放到鎖里面:

num = 1;
      total = 0;
      using (var q = new BlockingCollection<int>())
      {
        list.AsParallel().ForAll(n =>
        {
          var c = random.Next(1, 50);
          Interlocked.Add(ref total, c);
          for (int i = 0; i < c; i++)
          {
            
            // Task.Delay(100);
            q.Add(Interlocked.Increment(ref num));
            
            //可控
            //lock (objLock)
            //{
            //  num++;
            //  q.Add(num);
            //}
          }

        });
        q.CompleteAdding();
        Console.WriteLine($"num累計值為:{total},并發(fā)之后值為:{num}");
        var x = q.GroupBy(n => n).Where(o => o.Count() > 1);
        Console.WriteLine($"并發(fā)使用安全集合BlockingCollection+Interlocked添加num,集合重復值:{x.Count()}個");
        Console.ReadKey();
      }

這里我測試了另外一個線程安全的集合BlockingCollection,關于這個集合的使用請自行查找MSDN文檔,上面的關鍵代碼直接添加安全集合的返回值,可以保證集合不會重復,但其實下面的lock更適用與正式環(huán)境,因為我們添加的一般都是對象不會是基礎類型數值,運行結果如下:

至此,我們的問題解決了,計算時間由原來的9分多降至110秒左右,可見Parallel的處理還是很給力的,唯一不足的是,很占CPU,執(zhí)行計算后CPU達到了88%。附上計算結果:

優(yōu)化前后對比

總結:

C#安全集合在并發(fā)的情況下其實不一定是安全的,還是需要結合實際應用場景和驗證結果為準。Parallel.ForEach在對循環(huán)數量可觀的情況下是可以去使用的,如果有共享變量,一定要配合鎖做同步處理。還是得慎用這個方法,如果方法內部有操作數據庫的記得增加事務處理,否則就呵呵了。

好了,以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,謝謝大家對腳本之家的支持。

相關文章

  • asp.net獲取系統(tǒng)當前時間的方法詳解

    asp.net獲取系統(tǒng)當前時間的方法詳解

    這篇文章主要介紹了asp.net獲取系統(tǒng)當前時間的方法,較為詳細的分析了C#日期與時間操作所涉及的相關函數與使用技巧,需要的朋友可以參考下
    2016-06-06
  • C#實現(xiàn)加密與解密詳解

    C#實現(xiàn)加密與解密詳解

    本文詳細講解了C#實現(xiàn)加密與解密詳解的方法,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2022-06-06
  • 淺談C#跨線程調用窗體控件(比如TextBox)引發(fā)的線程安全問題

    淺談C#跨線程調用窗體控件(比如TextBox)引發(fā)的線程安全問題

    下面小編就為大家分享一篇淺談C#跨線程調用窗體控件(比如TextBox)引發(fā)的線程安全問題,具有很好的參考價值,希望對大家有所幫助
    2017-11-11
  • Unity3D實現(xiàn)攻擊范圍檢測

    Unity3D實現(xiàn)攻擊范圍檢測

    這篇文章主要為大家詳細介紹了Unity3D實現(xiàn)攻擊范圍檢測,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-04-04
  • WPF實現(xiàn)抽屜菜單效果的示例代碼

    WPF實現(xiàn)抽屜菜單效果的示例代碼

    這篇文章主要介紹了如何利用WPF實現(xiàn)抽屜菜單效果,文中的示例代碼講解詳細,對我們學習或工作有一定幫助,需要的可以參考一下
    2022-08-08
  • C#使用Consul集群進行服務注冊與發(fā)現(xiàn)

    C#使用Consul集群進行服務注冊與發(fā)現(xiàn)

    這篇文章主要介紹了C#使用Consul集群進行服務注冊與發(fā)現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2019-12-12
  • C#中內聯(lián)函數的用法介紹

    C#中內聯(lián)函數的用法介紹

    這篇文章介紹了C#中內聯(lián)函數的用法,文中通過示例代碼介紹的非常詳細。對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-03-03
  • C#使用Winform編寫一個圖片預覽器管理

    C#使用Winform編寫一個圖片預覽器管理

    這篇文章主要為大家詳細介紹了C#如何使用Winform編寫一個通用圖片預覽器管理,包含滾輪放大縮小,剪切,下一頁,方向變化等,需要的可以參考下
    2024-02-02
  • KMP算法的C#實現(xiàn)方法

    KMP算法的C#實現(xiàn)方法

    這篇文章主要介紹了KMP算法的C#實現(xiàn)方法,代碼簡潔實用,需要的朋友可以參考下
    2014-09-09
  • 詳解Unity中的ShaderGraph入門使用教程

    詳解Unity中的ShaderGraph入門使用教程

    Unity2018版本之后推出了一個可編程渲染管線工具ShaderGraph,讓我們可以通過可視化界面拖拽來實現(xiàn)著色器的創(chuàng)建和編輯,今天重點給大家介紹Unity中的ShaderGraph入門使用教程,需要的朋友參考下吧
    2021-07-07

最新評論