歡迎光臨
每天分享高質量文章

.net core 併發下的執行緒安全問題

抱歉,其實內容並不如題!!!

背景(寫測試demo所出現的異常,供大家學習與拍磚):

.net core webapi專案,做了一個授權的filter(真正的生產專案的話,JWT很棒),單個介面測試沒有問題,當用前端在同一個頁面呼叫多個介面的時候,執行服務,開啟頁面,然後……Exceptions……(真正的開發中大家應該也會遇到)

異常1:An attempt was made to use the context while it is being configured. A DbContext instance cannot be used inside OnConfiguring since it is still being configured at this point. This can happen if a second operation is started on this context before a previous operation completed. Any instance members are not guaranteed to be thread safe.

異常2:A second operation started on this context before a previous operation completed. Any instance members are not guaranteed to be thread safe.

異常3:Invalid attempt to call Read when reader is closed.

異常4:Unable to cast object of type ‘System.Data.ProviderBase.DbConnectionClosedConnecting’ to type ‘System.Data.SqlClient.SqlInternalConnectionTds’.

異常5:Object reference not set to an instance of an object.

異常6:不允許啟動新事務,因為有其他執行緒正在該會話中執行。

異常7:An error occurred while updating the entries. See the inner exception for details.

嘗試運行了N多遍,嗯,挺不穩定的(程式碼垃圾!),那看看異常吧

一看很容易理解:在前一個操作完成之前,在此背景關係中啟動第二個操作。任何實體成員都不能保證是執行緒安全的。就是說,我在用這個背景關係的時候,你來搶個屁……

這個可能發生在併發的情況下,同時使用了同一個背景關係……那麼開啟一個頁面,為什麼會同時使用同一個背景關係呢?好吧,在這裡要負荊請罪了(可以說是自己的問題)

我在Filter裡面有查詢,用到資料庫背景關係 。罪過咯,直接想在Filter裡面過濾黑名單,所以查了資料庫(這個業務是不合理的,這是一個作死的行為,請謹慎看待,這裡做學習討論之用)。

public class AuthFilterAttribute : ActionFilterAttribute
{

 public override void OnActionExecuting(ActionExecutingContext context)
       {

            base.OnActionExecuting(context);
               
            .....
              
            //判斷是否在黑名單內
            var blackList = _app.GetBlackList();
            ......
       }
}

這裡為什麼用 ActionFilterAttribute ?是因為測試的時候要監測一下介面執行的整個過程,So……

然後還有一些錯是:物件取用未設定為物件的實體。這個錯誤太常見,不就是物件為Null了嗎?但是,未實體化物件在業務邏輯上的情況太多了。我的應該有:

1、沒有獲取到當前物件,這是.net core,不是.net,不是因為沒有new物件。是註入中沒有註入成功,獲取註入後,沒有獲取到。(但我本來執行的好好的,是因為一下是開啟對接的頁面才發生的問題,可以排除了)

2、本來已經實體的物件被回收了……(這可能性嘛……有一定的可能,但發生在哪呢?)

找啊找,其實方向有了,但是自己卻沒想起來……

其實如果不確定的話,倒是可以先找找別人是怎麼說的(不是為了裝X,找開發上的問題我是推薦 github 和 stackoverflow 的,大部分的問題都可以找到):

(1)異常 1 還有同樣 一條搜尋結果

 

(2)異常 2

 

雖然以上找的不一定是真正的答案,至少提供了一個方向,並且你至少可以嘗試性地去解決一下。這裡提供的方向其實很明確:

1、是否應該使用 Scoped 和 Transient 的,你卻使用了 Singleton;

2、多執行緒中使用了 async 卻沒有配對的使用 await;

至少我找到的關鍵點是這兩個。

那怎麼找到並解決這個問題呢,.net core都是註入的,當然 AuthFilterAttribute 也是註入的。跑到 Startup一看,很明顯,問題出在哪裡了 — 單例!本應該是Scoped樣式的,卻用了單例。

那就將 AuthFilterAttribute 換一種註入樣式就行啦。

 

改為

 

我使用的是Filter,Filter有自己的生命週期,去確認一下:Filter的官方檔案

看到一張圖!!!(當然你也可以細細研讀一下這個檔案)如下:

這還不明顯?!!!

Filter會被回收的!!!這同樣解釋了 異常3、4、5、6、7所發生的原因。

OK,問題已經解決了,這是在開發中遇到的問題,可以說是涉及到.net core 本身的執行機制。

我算是一個應用型的程式員,喜歡在應用中學習底層的東西。那麼接下來當然就可以擴充套件 Singleton、Scoped 和 Transient 等知識了。

如不喜,請拍!

已同步到看一看
贊(0)

分享創造快樂