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

在Parallel中使用DbSet.Add()發(fā)現(xiàn)的一系列多線程問題和解決思路詳解

 更新時間:2016年11月14日 15:29:11   作者:balahoho  
這篇文章主要介紹了在Parallel中使用DbSet.Add()發(fā)現(xiàn)的一系列多線程問題和解決過程的相關資料,非常不錯,具有參考借鑒價值,需要的朋友可以參考下

發(fā)現(xiàn)問題

需求很簡單,大致就是要批量往數(shù)據(jù)庫寫數(shù)據(jù),于是打算用Parallel并行的方式寫入,希望能利用計算機多核特性加快程序執(zhí)行速度。想的很美好,于是快速擼了類似下面的一串代碼:

using (var db = new SmsEntities())
{
Parallel.For(0, 1000, (i) =>
{
db.MemberCard.Add(new MemberCard()
{
CardNo = "NO_" + i.ToString(),
Banlance = 0,
CreateTime = DateTime.Now,
Name = "Test_" + i.ToString(),
Status = 1
});
});
db.SaveChanges();
}

可意外的是竟然無情的報錯了:

奇葩的是當我再次刷新的時候異常又不一樣了,于是連著刷新好多次,總結出現(xiàn)過的異常有下面這些:

1、 未將對象引用設置到對象的實例。

2、 已添加了具有相同鍵的項。

3、 集合已修改;可能無法執(zhí)行枚舉操作。

4、 一個 EdmType 不能多次映射到 CLR 類。EdmType“SmsModel.MemberCard”映射了一次以上。

其中1和2是出現(xiàn)最多的,而且所有異常都是出現(xiàn)在Add的時候,各種吃瓜表情~沒辦法,接著一一斷點調試,還是沒找出原因,出于進度考慮,換成了另一種方案,也就是用DbSet的AddRange方法。先在Parallel中累加出一個實體List,然后一次性添加到DbSet中,代碼演變?yōu)?

List<MemberCard> list = new List<MemberCard>();
using (var db = new SmsEntities())
{
var result = Parallel.For(0, 1000, (i) =>
{
list.Add(new MemberCard()
{
CardNo = "NO_" + i.ToString(),
Banlance = 0,
CreateTime = DateTime.Now,
Name = "Test_" + i.ToString(),
Status = 1
});
});
if (result.IsCompleted)
{
db.MemberCard.AddRange(list);
db.SaveChanges();
}
}

然后編譯、測試,沒問題,就先放著了。

分析問題

第二天到公司心里還在糾結這個問題,于是打開頁面輸入生成的數(shù)據(jù)量1000(真實項目中的循環(huán)次數(shù)是手動輸入的),點按鈕提交,嗯,又吃瓜般的異常了…:

心想昨天測試都好好的?。ㄆ鋵嵶蛱燧斎氲氖?0,心虛臉...),沒辦法,上斷點吧,一看嚇一跳:

明明循環(huán)1000次,結果只有971條數(shù)據(jù),而且里面還有為null的,經(jīng)過多次調試發(fā)現(xiàn)這是一個隨機現(xiàn)象,Count是隨機的null也是隨機的,有時出現(xiàn)有時沒有,初步判斷這是一個在多線程情況下引發(fā)的一個資源調配異常。So,上MSDN看了一下List的介紹,最后面“線程安全”寫著:

一切貌似都清楚了,于是打算驗證一下結果,加上了鎖,測試結果為:

list里面也沒有再出現(xiàn)null了,確認是因為多線程安全引起的異常。于是想起昨天那個問題是否也是同樣的問題,再上MSDN搜了一下DbContext類和DbSet類,都是這樣說的:

接著就給dbcontext上了鎖,測試,這次總算如我所料,完美運行。但是不解的是最初那幾個異常是如何產生的,List中雖然數(shù)量不夠也存在為null的對象,但是并沒有直接爆出異?!,F(xiàn)在只知道是線程問題,再詳細的也搞不清楚,有知道的大神還麻煩指點一下。

尋找解決方案并驗證結論

也想過用Partitioner分區(qū)來做,但是仔細一想,雖然分區(qū)內部是單線程,但是區(qū)與區(qū)之間還是多線程的,如果分的太細也就失去了Parallel的意義,只得另尋出路。還好Framework為我們也提供了一些線程安全的泛型集合(比如ConcurrentBag、ConcurrentQueue等),不過其本質還是用了鎖,于是就綜合做了一下單線程list、多線程list加鎖、多線程ConcurrentBag、多線程ConcurrentQueue的性能對比,結果如下:

循環(huán)1000次時:

循環(huán)10000次時:

循環(huán)100000次時:

得出結論就是,在執(zhí)行次數(shù)超大時用線程安全類型會更慢,在執(zhí)行次數(shù)較少時線程安全類型也沒什么優(yōu)勢。

解決問題

最后在經(jīng)過仔細測試驗證和考慮項目實際需求(幾乎不可能一次10000)后,去繁從簡,回歸原始,用最簡單直白的寫法單線程循環(huán)來完成。雖然一番折騰下來還是回到最初,但是這過程中讓我發(fā)現(xiàn)了意料之外問題,然后找到了原因,然后測試驗證,最終得到了最優(yōu)解決方案。還是那句話,填完坑,你就比之前更強大了!

相關文章

  • Unity技術手冊之Slider滑動器使用實例詳解

    Unity技術手冊之Slider滑動器使用實例詳解

    這篇文章主要為大家介紹了Unity技術手冊之Slider滑動器使用實例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-11-11
  • c#基礎——了解程序結構

    c#基礎——了解程序結構

    這篇文章主要介紹了c# 程序結構的相關資料,文中講解非常細致,幫助大家更好的理解和學習C#,感興趣的朋友可以了解下
    2020-07-07
  • C# 如何設置label(標簽)控件的背景顏色為透明

    C# 如何設置label(標簽)控件的背景顏色為透明

    這篇文章主要介紹了C# 如何設置label(標簽)控件的背景顏色為透明,幫助大家更好的理解和使用c#,感興趣的朋友可以了解下
    2020-10-10
  • C# 委托的三種調用示例(同步調用 異步調用 異步回調)

    C# 委托的三種調用示例(同步調用 異步調用 異步回調)

    本文將主要通過同步調用、異步調用、異步回調三個示例來講解在用委托執(zhí)行同一個加法類的時候的的區(qū)別和利弊
    2013-12-12
  • C#提取網(wǎng)頁中超鏈接link和text部分的方法

    C#提取網(wǎng)頁中超鏈接link和text部分的方法

    這篇文章主要介紹了C#提取網(wǎng)頁中超鏈接link和text部分的方法,涉及C#正則表達式及字符串操作相關技巧,需要的朋友可以參考下
    2016-02-02
  • C#實現(xiàn)FFT(遞歸法)的示例代碼

    C#實現(xiàn)FFT(遞歸法)的示例代碼

    FFT是數(shù)字信號處理中的重要算法。這篇文章將為大家詳細介紹一下如何利用C#語言實現(xiàn)FFT(遞歸法),文中的示例代碼講解詳細,感興趣的可以了解一下
    2022-07-07
  • C#實現(xiàn)全局快捷鍵功能

    C#實現(xiàn)全局快捷鍵功能

    這篇文章介紹了C#實現(xiàn)全局快捷鍵功能的方法,文中通過示例代碼介紹的非常詳細。對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-06-06
  • C#中函數(shù)的創(chuàng)建和閉包的理解

    C#中函數(shù)的創(chuàng)建和閉包的理解

    這篇文章主要介紹了C#中函數(shù)的創(chuàng)建和閉包的理解,本文講解了動態(tài)創(chuàng)建函數(shù)、匿名函數(shù)不足之處、理解c#中的閉包、閉包的優(yōu)點等內容,需要的朋友可以參考下
    2015-04-04
  • 詳解C#如何利用TcpListener和TcpClient實現(xiàn)Tcp通訊

    詳解C#如何利用TcpListener和TcpClient實現(xiàn)Tcp通訊

    TcpListener 和 TcpClient 是在 System.Net.Sockets.Socket 類的基礎上做的進一步封裝,使用 GetStream 方法返回網(wǎng)絡流,下面我們就來詳細一下如何使用TcpListener和TcpClient實現(xiàn)Tcp通訊吧
    2023-12-12
  • C#保存與讀取DataTable信息到XML格式的方法

    C#保存與讀取DataTable信息到XML格式的方法

    這篇文章主要介紹了C#保存與讀取DataTable信息到XML格式的方法,實例分析了C#讀取DataTable信息到XML格式及讀取XML格式數(shù)據(jù)到DataTable的相關技巧,具有一定參考借鑒價值,需要的朋友可以參考下
    2015-08-08

最新評論