主頁 > .NET開發 > 編碼技巧 --- 同步鎖物件的選定

編碼技巧 --- 同步鎖物件的選定

2023-07-13 08:37:03 .NET開發

引言

在C#中,讓執行緒同步有兩種方式:

  • 鎖(lock、Monitor)
  • 信號量(EventWaitHandle、Semaphore、Mutex)

執行緒鎖的原理,就是鎖住一個資源,使得應用程式在此刻只有一個執行緒訪問該資源,通俗地講,就是讓多執行緒變成單執行緒,在C#中,可以將被鎖定的資源理解成 new 出來的普通CLR物件,

如何選定

既然需要鎖定的資源就是C#中的一個物件,我們就該仔細思考,到底什么樣的物件能夠成為一個鎖物件(也叫同步物件)?

那么選擇同步物件的時候,應當始終注意以下幾點:

  1. 同步物件在需要同步的多個執行緒中是可見的同一個物件,
  2. 在非靜態方法中,靜態變數不應作為同步物件,
  3. 值型別物件不能作為同步物件,
  4. 避免將字串作為同步物件,
  5. 降低同步物件的可見性,

原因分析

接下來就探討一下這五種情況,

注意事項1:需要鎖定的物件在多個執行緒中是可見的,而且是同一個物件,

“可見的”這是顯而易見的,如果物件不可見,就不能被鎖定,

“同一個物件”,這也很容易理解,如果鎖定的不是同一個物件,那又如何來同步兩個物件呢?

雖然理解起來簡單,但不見得我們在這上面就不會犯錯誤,

我們模擬一個必須使用到鎖的場景:在遍歷一個集合的程序中,同時在另外一個執行緒中洗掉集合中的某項,

下面這個例子中,如果沒有 lock 陳述句,將會拋出例外System.InvalidOperationException:“Collection was modified; enumeration operation may not execute.”

public partial class Form1 : Form
{
    public Form1()
    {
        InitializeComponent();
    }

    AutoResetEvent autoResetEvent = new AutoRe

    List<string> strings = new List<string>() 

    private void btn_StartThreads_Click(object
    {
        object syncObj = new object();

        Thread t1 = new Thread(() =>
        {
            //確保等待t2開始之后才運行下面的代碼
            autoResetEvent.WaitOne();

            lock (syncObj)
            {
                foreach (var item in strings)
                {
                    Thread.Sleep(1000);
                }
            }
        });
        t1.IsBackground = false;

        t1.Start();

        Thread t2 = new Thread(() =>
        {
            autoResetEvent.Set();

            Thread.Sleep(1000);

            lock (syncObj)
            {
                strings.RemoveAt(1);
            }

        });
        t2.IsBackground = false;

        t2.Start();
    }
}

上述例子是 Winform 表單應用程式,按鈕的單擊事件中演示該功能,物件 syncObj 對于執行緒 t1t2 來說,在CLR中肯定是同一個物件,所以,上面的示例運行是沒有問題的,

現在,我們將此示例重構,將實際的作業代碼移到一個型別 SampleClass 中,該示例要在多個 SampleClass 實體間操作一個靜態欄位,如下所示:

public partial class Form1 : Form
{
    public Form1()
    {
        InitializeComponent();
    }

    private void btn_StartThreads_Click(object sender, EventArgs e)
    {
        SampleClass sampleClass1 = new SampleClass();
        SampleClass sampleClass2 = new SampleClass();
        sampleClass1.StartT1();
        sampleClass2.StartT2();
    }
}

public class SampleClass
{
    public static AutoResetEvent autoResetEvent = new AutoResetEvent(false);

    static List<string> strings = new List<string>() { "str1", "str2", "str3" };

    object syncObj = new object();

    public void StartT1()
    {
        Thread t1 = new Thread(() =>
        {
            //確保等待t2開始之后才運行下面的代碼
            autoResetEvent.WaitOne();

            lock (syncObj)
            {
                foreach (var item in strings)
                {
                    Thread.Sleep(1000);
                }
            }
        });
        t1.IsBackground = false;

        t1.Start();
    }
    public void StartT2()
    {
        Thread t2 = new Thread(() =>
        {
            autoResetEvent.Set();

            Thread.Sleep(1000);

            lock (syncObj)
            {
                strings.RemoveAt(1);
            }

        });
        t2.IsBackground = false;

        t2.Start();
    }
}

該例子運行起來就會拋出例外System.InvalidOperationException:“Collection was modified; enumeration operation may not execute.”

查看型別 SampleClass 的方法 StartT1StartT2 ,方法內部鎖定的是 SampleClass 的實體變數 syncObj

實體變數意味著,每創建一個 SampleClass 的實體都會生成一個 syncObj 物件,

在本例中,呼叫者一共創建了兩個 SampleClass 實體,繼而分別呼叫:

samplel.StartTl();
sample2.StartT2();

也就是說,以上代碼鎖定的是兩個不同的 syncObj ,這等于完全沒有達到兩個執行緒鎖定同一個物件的目的,

要修正以上錯誤,只要將 syncObj 變成 static 就可以了,

另外,思考一下 lock(this) ,我們同樣不建議在代碼中撰寫這樣的代碼,如果兩個物件的實體分別執行了鎖定的代碼,實際鎖定的也就會是兩個物件,完全不能達到同步的目的,

第二個注意事項:在非靜態方法中,靜態變數不應作為同步物件,

上文說到,要修正第一個注意事項中的示例問題,需要將 syncObj 變成 static,這似乎和本注意事項有矛盾,事實上,第一個注意事項中的示例代碼僅僅出于演示的目的,在實際應用中,我們非常不建議撰寫此類代碼,

在撰寫多執行緒代碼時,要遵循這樣的一個原則:

型別的靜態方法應當保證執行緒安全,非靜態方法不需實作執行緒安全,

FCL中的絕大部分類都遵循了這個原則,

像上一個示例中,如果將 syncObj 變成 static,就相當于讓非靜態方法具備了執行緒安全性,這帶來的一個問題是,如果應用程式中該型別存在多個實體,在遇到這個鎖的時候,它們都會產生同步,而這可能不是開發者所愿意看到的,第二個注意事項實際也可以歸納到第一個注意事項中,

第三個注意事項:值型別物件不能作為同步物件,

值型別在傳遞到另一個執行緒的時候,會創建一個副本,這相當于每個執行緒鎖定的也是兩個物件,因此,值型別物件不能作為同步物件,
第四個注意事項:鎖定字串是完全沒有必要的,而且相當危險,

這整個程序看上去和值型別正好相反,字串在CLR中會被暫存到記憶體里,如果有兩個變數被分配了相同內容的字串,那么這兩個參考會被指向同一塊記憶體,所以,如果有兩個地方同時使用了lock(“abc”) ,那么它們實際鎖定的是同一個物件,這會導致整個應用程式被阻滯,

第五個注意事項:降低同步物件的可見性,

可見范圍最廣的一種同步物件是 typeof(SampleClass)

typeof()方法所回傳的結果(也就是型別的type)是SampleClass 的所有實體所共有的,即:所有實體的type都指向typeof方法的結果,

這樣一來,如果我們 lock(typeof(SampleClass) ,當前應用程式中所有 SampleClass 的實體執行緒將會全部被同步,這樣編碼完全沒有必要,而且這樣的同步物件太開放了,

一般來說,同步物件也不應該是一個公共變數或屬性,在FCL的早期版本中,一些常用的集合型別(如 ArrayList )提供了公共屬性 SyncRoot ,讓我們鎖定以便進行一些執行緒安全的操作,

所以你一定會覺得我們剛才的結論不正確,其實不然,ArrayList 操作的大部分應用場景不涉及多執行緒同步,所以它的方法更多的是單執行緒應用場景,執行緒同步是一個非常耗時(低效)的操作,若 ArrayList 的所有非靜態方法都要考慮執行緒安全,那么 ArrayList 完全可以將這個 SyncRoot 變成靜態私有的,現在它將 SyncRoot 變為公開的,是讓呼叫者自己去決定操作是否需要執行緒安全,

我們在撰寫代碼時,除非有這樣的要求,否則就應該始終考慮降低同步物件的可見性,將同步物件藏起來,只開放給自己或自己的子類就夠了(需要開放給子類的情況其實也不多),

本篇內容參考自

撰寫高質量代碼:改善C#程式的157個建議 / 陸敏技著.一北京:機械工業出版社,2011.9

作者: Niuery Daily

出處: https://www.cnblogs.com/pandefu/>

郵箱: [email protected]

關于作者:.Net Framework,.Net Core ,WindowsForm,WPF ,控制元件庫,多執行緒

本文著作權歸作者所有,歡迎轉載,但未經作者同意必須保留此段宣告,且在文章頁面明顯位置給出 原文鏈接,否則保留追究法律責任的權利, 如有問題, 可郵件咨詢,

轉載請註明出處,本文鏈接:https://www.uj5u.com/net/557181.html

標籤:.NET技术

上一篇:編碼技巧 --- 同步鎖物件的選定

下一篇:返回列表

標籤雲
其他(162504) Python(38274) JavaScript(25532) Java(18294) C(15242) 區塊鏈(8275) C#(7972) AI(7469) 爪哇(7425) MySQL(7299) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5876) 数组(5741) R(5409) Linux(5347) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4616) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2439) ASP.NET(2404) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) HtmlCss(2002) .NET技术(1988) 功能(1967) Web開發(1951) C++(1942) python-3.x(1918) 弹簧靴(1913) xml(1889) PostgreSQL(1884) .NETCore(1863) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • WebAPI簡介

    Web體系結構: 有三個核心:資源(resource),URL(統一資源識別符號)和表示 他們的關系是這樣的:一個資源由一個URL進行標識,HTTP客戶端使用URL定位資源,表示是從資源回傳資料,媒體型別是資源回傳的資料格式。 接下來我們說下HTTP. HTTP協議的系統是一種無狀態的方式,使用請求/ ......

    uj5u.com 2020-09-09 22:07:47 more
  • asp.net core 3.1 入口:Program.cs中的Main函式

    本文分析Program.cs 中Main()函式中代碼的運行順序分析asp.net core程式的啟動,重點不是剖析原始碼,而是理清程式開始時執行的順序。到呼叫了哪些實體,哪些法方。asp.net core 3.1 的程式入口在專案Program.cs檔案里,如下。ususing System; us ......

    uj5u.com 2020-09-09 22:07:49 more
  • asp.net網站作為websocket服務端的應用該如何寫

    最近被websocket的一個問題困擾了很久,有一個需求是在web網站中搭建websocket服務。客戶端通過網頁與服務器建立連接,然后服務器根據ip給客戶端網頁發送資訊。 其實,這個需求并不難,只是剛開始對websocket的內容不太了解。上網搜索了一下,有通過asp.net core 實作的、有 ......

    uj5u.com 2020-09-09 22:08:02 more
  • ASP.NET 開源匯入匯出庫Magicodes.IE Docker中使用

    Magicodes.IE在Docker中使用 更新歷史 2019.02.13 【Nuget】版本更新到2.0.2 【匯入】修復單列匯入的Bug,單元測驗“OneColumnImporter_Test”。問題見(https://github.com/dotnetcore/Magicodes.IE/is ......

    uj5u.com 2020-09-09 22:08:05 more
  • 在webform中使用ajax

    如果你用過Asp.net webform, 說明你也算是.NET 開發的老兵了。WEBform應該是2011 2013左右,當時還用visual studio 2005、 visual studio 2008。后來基本都用的是MVC。 如果是新開發的專案,估計沒人會用webform技術。但是有些舊版 ......

    uj5u.com 2020-09-09 22:08:50 more
  • iis添加asp.net網站,訪問提示:由于擴展配置問題而無法提供您請求的

    今天在iis服務器配置asp.net網站,遇到一個問題,記錄一下: 問題:由于擴展配置問題而無法提供您請求的頁面。如果該頁面是腳本,請添加處理程式。如果應下載檔案,請添加 MIME 映射。 WindowServer2012服務器,添加角色安裝完.netframework和iis之后,運行aspx頁面 ......

    uj5u.com 2020-09-09 22:10:00 more
  • WebAPI-處理架構

    帶著問題去思考,大家好! 問題1:HTTP請求和回傳相應的HTTP回應資訊之間發生了什么? 1:首先是最底層,托管層,位于WebAPI和底層HTTP堆疊之間 2:其次是 訊息處理程式管道層,這里比如日志和快取。OWIN的參考是將訊息處理程式管道的一些功能下移到堆疊下端的OWIN中間件了。 3:控制器處理 ......

    uj5u.com 2020-09-09 22:11:13 more
  • 微信門戶開發框架-使用指導說明書

    微信門戶應用管理系統,采用基于 MVC + Bootstrap + Ajax + Enterprise Library的技術路線,界面層采用Boostrap + Metronic組合的前端框架,資料訪問層支持Oracle、SQLServer、MySQL、PostgreSQL等資料庫。框架以MVC5,... ......

    uj5u.com 2020-09-09 22:15:18 more
  • WebAPI-HTTP編程模型

    帶著問題去思考,大家好!它是什么?它包含什么?它能干什么? 訊息 HTTP編程模型的核心就是訊息抽象,表示為:HttPRequestMessage,HttpResponseMessage.用于客戶端和服務端之間交換請求和回應訊息。 HttpMethod類包含了一組靜態屬性: private stat ......

    uj5u.com 2020-09-09 22:15:23 more
  • 部署WebApi隨筆

    一、跨域 NuGet參考Microsoft.AspNet.WebApi.Cors WebApiConfig.cs中配置: // Web API 配置和服務 config.EnableCors(new EnableCorsAttribute("*", "*", "*")); 二、清除默認回傳XML格式 ......

    uj5u.com 2020-09-09 22:15:48 more
最新发布
  • 編碼技巧 --- 同步鎖物件的選定

    <a href="https://www.cnblogs.com/pandefu/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/1630815/20190315181946.png" alt="" /><...

    uj5u.com 2023-07-13 08:37:03 more
  • 編碼技巧 --- 同步鎖物件的選定

    ## 引言 在C#中,讓執行緒同步有兩種方式: - 鎖(lock、Monitor) - 信號量(EventWaitHandle、Semaphore、Mutex) 執行緒鎖的原理,就是鎖住一個資源,使得應用程式在此刻只有一個執行緒訪問該資源。通俗地講,就是讓多執行緒變成單執行緒。在C#中,可以將被鎖定的資源理解 ......

    uj5u.com 2023-07-13 08:36:31 more
  • 編碼技巧 --- 如何實作字串運算運算式的計算

    <a href="https://www.cnblogs.com/pandefu/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/1630815/20190315181946.png" alt="" /><...

    uj5u.com 2023-07-12 08:56:02 more
  • 編碼技巧 --- 如何實作字串運算運算式的計算

    ## 引言 最近做一個配置的功能,需求是該配置項跟另一個整形配置項關聯,具有一定的函式關系,例如有一個配置項是值為 `N` ,則另一配置 `F` 項滿足函式關系$F=2/(N+1)$。這個函式關系是客戶手動輸入,只需要簡單的四則運算,所以我們要做的就是判斷四則運算運算式是否有效,且給定 `N` 的值 ......

    uj5u.com 2023-07-12 08:55:12 more
  • 玻璃窯爐入爐配料稱重從硬體模型搭建到稱重資料寫表

    北辰模塊 IPv4 地址: 10.30.15.244IPv4 子網掩碼: 255.255.255.0 IPv4 默認網關: 10.30.15.254 串口服務器 ......

    uj5u.com 2023-07-07 09:10:05 more
  • 玻璃窯爐入爐配料稱重從硬體模型搭建到稱重資料寫表

    北辰模塊 IPv4 地址: 10.30.15.244IPv4 子網掩碼: 255.255.255.0 IPv4 默認網關: 10.30.15.254 串口服務器 ......

    uj5u.com 2023-07-07 09:09:34 more
  • Prism導航

    > 通常,導航意味著某個Control被添加到UI中,與此同時另一個Control被移除。 # 簡單跳轉 1. 新建 `UserControl`,新建ViewModel,VM需要實作 `INavigationAware` 2. 注冊 `UserControl`到DryIoc容器 ``` contai ......

    uj5u.com 2023-06-26 09:37:26 more
  • Prism導航

    > 通常,導航意味著某個Control被添加到UI中,與此同時另一個Control被移除。 # 簡單跳轉 1. 新建 `UserControl`,新建ViewModel,VM需要實作 `INavigationAware` 2. 注冊 `UserControl`到DryIoc容器 ``` contai ......

    uj5u.com 2023-06-26 09:36:38 more
  • .netcore中的虛擬檔案EmbeddedFile

    以前一直比較好奇像swagger,cap,skywalking等組件是如何實作參考一個dll即可在網頁上展示界面的,難道這么多html,js,css等都是硬編碼寫死在代碼檔案中的?后面接觸apb里面也有虛擬檔案的功能,一直沒去深入了解,最近仔細看了一下他們的代碼,發現內部其實就是用嵌入式檔案(Emb ......

    uj5u.com 2023-06-08 09:47:43 more
  • .netcore中的虛擬檔案EmbeddedFile

    以前一直比較好奇像swagger,cap,skywalking等組件是如何實作參考一個dll即可在網頁上展示界面的,難道這么多html,js,css等都是硬編碼寫死在代碼檔案中的?后面接觸apb里面也有虛擬檔案的功能,一直沒去深入了解,最近仔細看了一下他們的代碼,發現內部其實就是用嵌入式檔案(Emb ......

    uj5u.com 2023-06-08 09:46:11 more