C#中Lock的秘密
一、概要
本文主要講解在c#中l(wèi)ock關(guān)鍵字的用法以及需要注意的坑。幫助大家避免使用不當(dāng)造成的bug。
作用:lock 關(guān)鍵字可以用來確保代碼塊完成運(yùn)行,而不會(huì)被其他線程中斷。它可以把一段代碼定義為互斥段(critical p),互斥段在一個(gè)時(shí)刻內(nèi)只允許一個(gè)線程進(jìn)入執(zhí)行,而其他線程必須等待。這是通過在代碼塊運(yùn)行期間為給定對象獲取互斥鎖來實(shí)現(xiàn)的。在多線程中,每個(gè)線程都有自己的資源,但是代碼區(qū)是共享的,即每個(gè)線程都可以執(zhí)行相同的函數(shù)。這可能帶來的問題就是幾個(gè)線程同時(shí)執(zhí)行一個(gè)函數(shù),導(dǎo)致數(shù)據(jù)的混亂,產(chǎn)生不可預(yù)料的結(jié)果,因此我們必須避免這種情況的發(fā)生。
缺點(diǎn): 多線程中頻繁使用lock會(huì)造成性能損耗。
二、詳細(xì)內(nèi)容
(1)使用
以下是lock在單例中使用的,大家可以看到在Instance中有兩個(gè)if判斷_instance是否為空。為什么?因?yàn)閘ock在執(zhí)行的過程中會(huì)有性能損耗如果已經(jīng)初始化過了之后就不要在走lock加鎖了,多線程中只讀單例 對象是不會(huì)造成‘臟讀’數(shù)據(jù)的。那么最外層的if就完美避免了lock的缺點(diǎn)。
public class Demo1 {private static readonly object _lockObj = new object();private static Demo1 _instance;public static Demo1 Instance{get{if (_instance == null){lock (_lockObj){if (_instance == null){_instance = new Demo1();}}}return _instance;}}private Demo1() { }public List<string> GetData() {return new List<string>();} }(2)注意事項(xiàng)及原理
2.1注意事項(xiàng)
當(dāng)同步對共享資源的線程訪問時(shí),請鎖定專用對象實(shí)例(例如,private readonly object balanceLock = new object();)或另一個(gè)不太可能被代碼無關(guān)部分用作 lock 對象的實(shí)例。避免對不同的共享資源使用相同的 lock 對象實(shí)例,因?yàn)檫@可能導(dǎo)致死鎖或鎖爭用。具體而言,避免將以下對象用作 lock 對象:
this(調(diào)用方可能將其用作 lock)。
Type 實(shí)例(可以通過 typeof 運(yùn)算符或反射獲取)。
字符串實(shí)例,包括字符串文本,(這些可能是暫存的)。
盡可能縮短持有鎖的時(shí)間,以減少鎖爭用。
在 lock 語句的正文中不能使用 await 運(yùn)算符。
2.2原理(以下內(nèi)容比較淺顯,太深究內(nèi)容一篇文章寫不完)
Q1:大家會(huì)注意到,為什么要在lock的圓括號(hào)里放一個(gè)引用類型object?為什么不可以放一個(gè)值類型例如int?
A1:因?yàn)槿绻褂昧酥殿愋屠鏸nt作為lock鎖定的對象,lock圓括號(hào)中的入?yún)⑹莖bject類型當(dāng)傳入了值類型會(huì)對傳入的對象類型進(jìn)行轉(zhuǎn)換,那么在IL層面會(huì)對值類型進(jìn)行一次裝箱(box)操作。那么這種情況下就不具備lock鎖定需要用到專用對象的穩(wěn)定性了。
IL_0002:ldloc.0 IL_0003:box [mscorlib]System.Int32A2:第二個(gè)原因這個(gè)就需要追溯到“值類型”和“引用類型”的基類,大家都知道引用類型的基類是object、值類型的基類是ValueType這兩種基類本質(zhì)的區(qū)別如下:
值類型:構(gòu)造中不包含同步塊索引。
引用類型:構(gòu)造中包含同步塊索引。
除了c#語法不支持以外它不適宜作為lock圓括號(hào)中的鎖定對象的原因就是沒有同步塊索引。
總結(jié)
以上是生活随笔為你收集整理的C#中Lock的秘密的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: c#爬虫-解决ChromeDriver
- 下一篇: foreach和IEnumerable+