Parallel.ForEach怎么在C#項目中使用?針對這個問題,這篇文章詳細介紹了相對應(yīng)的分析和解答,希望可以幫助更多想解決這個問題的小伙伴找到更簡單易行的方法。
讓客戶滿意是我們工作的目標,不斷超越客戶的期望值來自于我們對這個行業(yè)的熱愛。我們立志把好的技術(shù)通過有效、簡單的方式提供給客戶,將通過不懈努力成為客戶在信息化領(lǐng)域值得信任、有價值的長期合作伙伴,公司提供的服務(wù)項目有:域名申請、虛擬主機、營銷軟件、網(wǎng)站建設(shè)、云巖網(wǎng)站維護、網(wǎng)站推廣。int num = 1; Listlist = new List (); 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次自增,正常結(jié)果應(yīng)該是2001,但實際結(jié)果如下:
有經(jīng)驗的同學(xué),立馬能想到需要加鎖了,C#內(nèi)置了很多鎖對象,如lock 互斥鎖,Interlocked 內(nèi)部鎖,Monitor 這幾個比較常見,lock內(nèi)部實現(xiàn)其實就是使用了Monitor對象。對變量自增,Interlocked對象提供了,變量自增,自減、或者相加等方法,我們使用自增方法Interlocked.Increment,函數(shù)定義為:int Increment(ref int num),該對象提供原子性的變量自增操作,傳入目標數(shù)值,返回或者ref num都是自增后的結(jié)果。 在之前的基礎(chǔ)上我們增加一些代碼:
num = 1; Console.WriteLine($"num初始值為:" + num.ToString()); list.AsParallel().ForAll(n => { Interlocked.Increment(ref num); }); Console.WriteLine($"使用內(nèi)部鎖,并發(fā){list.Count}次后為:" + num.ToString()); Console.ReadKey();
我們來看運行結(jié)果:
加了鎖之后ID重復(fù)算是解決了,其實別高興太早,由于正常的環(huán)境有了ID我們還有用這些ID來構(gòu)建對象呢,于是又寫了寫代碼,用集合來添加這些ID,為了更真實的模擬生產(chǎn)環(huán)境,我在forAll里面又加了一層循環(huán)代碼如下:
num = 1; Random random = new Random(); var total = 0; var m = new ConcurrentBag(); 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($"使用內(nèi)部鎖,并發(fā)+內(nèi)部循環(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,集合重復(fù)值:{l.Count()}個"); Console.ReadKey();
上面的代碼里面我用到了線程安全集合ConcurrentBag
num = 1; total = 0; using (var q = new BlockingCollection()) { 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,集合重復(fù)值:{x.Count()}個"); Console.ReadKey(); }
這里我測試了另外一個線程安全的集合BlockingCollection,關(guān)于這個集合的使用請自行查找MSDN文檔,上面的關(guān)鍵代碼直接添加安全集合的返回值,可以保證集合不會重復(fù),但其實下面的lock更適用與正式環(huán)境,因為我們添加的一般都是對象不會是基礎(chǔ)類型數(shù)值,運行結(jié)果如下:
至此,我們的問題解決了,計算時間由原來的9分多降至110秒左右,可見Parallel的處理還是很給力的,唯一不足的是,很占CPU,執(zhí)行計算后CPU達到了88%。附上計算結(jié)果:
優(yōu)化前后對比
關(guān)于Parallel.ForEach怎么在C#項目中使用問題的解答就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,如果你還有很多疑惑沒有解開,可以關(guān)注創(chuàng)新互聯(lián)成都網(wǎng)站設(shè)計公司行業(yè)資訊頻道了解更多相關(guān)知識。
另外有需要云服務(wù)器可以了解下創(chuàng)新互聯(lián)scvps.cn,海內(nèi)外云服務(wù)器15元起步,三天無理由+7*72小時售后在線,公司持有idc許可證,提供“云服務(wù)器、裸金屬服務(wù)器、高防服務(wù)器、香港服務(wù)器、美國服務(wù)器、虛擬主機、免備案服務(wù)器”等云主機租用服務(wù)以及企業(yè)上云的綜合解決方案,具有“安全穩(wěn)定、簡單易用、服務(wù)可用性高、性價比高”等特點與優(yōu)勢,專為企業(yè)上云打造定制,能夠滿足用戶豐富、多元化的應(yīng)用場景需求。