中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

為什么不要lock(this) ,鎖定變量最好是readonly

發布時間:2020-08-07 03:06:32 來源:網絡 閱讀:131 作者:森大科技 欄目:編程語言

來自森大科技官方博客
http://www.cnsendblog.com/index.php/?p=392
GPS平臺、網站建設、軟件開發、系統運維,找森大網絡科技!
http://cnsendnet.taobao.com

一. 為什么要lock,lock了什么?

當我們使用線程的時候,效率最高的方式當然是異步,即各個線程同時運行,其間不相互依賴和等待。但當不同的線程都需要訪問某個資源的時候,就需要同步機制了,也就是說當對同一個資源進行讀寫的時候,我們要使該資源在同一時刻只能被一個線程操作,以確保每個操作都是有效即時的,也即保證其操作的原子性。lock是C#中最常用的同步方式,格式為lock(objectA){codeB} 。

lock(objectA){codeB} 看似簡單,實際上有三個意思,這對于適當地使用它至關重要:

  1. objectA被lock了嗎?沒有則由我來lock,否則一直等待,直至objectA被釋放。
  2. lock以后在執行codeB的期間其他線程不能調用codeB,也不能使用objectA。
  3. 執行完codeB之后釋放objectA,并且codeB可以被其他線程訪問。

二. lock(this)怎么了?

我們看一個例子:

using System;
using System.Threading;

namespace Namespace1
{
    class C1
     {
        private bool deadlocked= true;

        //這個方法用到了lock,我們希望lock的代碼在同一時刻只能由一個線程訪問
        public void LockMe(object o)
         {
            lock (this)
             {
                while(deadlocked)
                 {
                     deadlocked = (bool)o;
                     Console.WriteLine("Foo: I am locked :(");
                     Thread.Sleep(500);
                 }
             }
         }

        //所有線程都可以同時訪問的方法
        public void DoNotLockMe()
         {
             Console.WriteLine("I am not locked :)");
         }
     }

    class Program
     {
        staticvoid Main(string[] args)
         {
             C1 c1 =new C1();

            //在t1線程中調用LockMe,并將deadlock設為true(將出現死鎖)
             Thread t1= new Thread(c1.LockMe);
             t1.Start(true);
             Thread.Sleep(100);

            //在主線程中lock c1
            lock (c1)
             {
                //調用沒有被lock的方法
                 c1.DoNotLockMe();
                //調用被lock的方法,并試圖將deadlock解除
                 c1.LockMe(false);
             }
         }
     }

在t1線程中,LockMe調用了lock(this), 也就是Main函數中的c1,這時候在主線程中調用lock(c1)時,必須要等待t1中的lock塊執行完畢之后才能訪問c1,即所有c1相關的操作都無法完成,于是我們看到連c1.DoNotLockMe()都沒有執行。

把C1的代碼稍作改動:

class C1
     {
        privatebool deadlocked= true;
        private object locker= new object();

        //這個方法用到了lock,我們希望lock的代碼在同一時刻只能由一個線程訪問
        public void LockMe(object o)
         {
            lock (locker)
             {
                while(deadlocked)
                 {
                     deadlocked = (bool)o;
                     Console.WriteLine("Foo: I am locked :(");
                     Thread.Sleep(500);
                 }
             }
         }

        //所有線程都可以同時訪問的方法
        public void DoNotLockMe()
         {
             Console.WriteLine("I am not locked :)");
         }
     }

這次我們使用一個私有成員作為鎖定變量(locker),在LockMe中僅僅鎖定這個私有locker,而不是整個對象。這時候重新運行程序,可以看到雖然t1出現了死鎖,DoNotLockMe()仍然可以由主線程訪問;LockMe()依然不能訪問,原因是其中鎖定的locker還沒有被t1釋放。

關鍵點:

  1. lock(this)的缺點就是在一個線程鎖定某對象之后導致整個對象無法被其他線程訪問。
  2. 鎖定的不僅僅是lock段里的代碼,鎖本身也是線程安全的。
  3. 我們應該使用不影響其他操作的私有對象作為locker。
  4. 在使用lock的時候,被lock的對象(locker)一定要是引用類型的,如果是值類型,將導致每次lock的時候都會將該對象裝箱為一個新的引用對象(事實上如果使用值類型,C#編譯器(3.5.30729.1)在編譯時就會給出一個錯誤)。

kenny add

而對于Monitor,發現它的靜態方法Enter(object obj)有一個異常類型ArgumentNullException,
執行lock(null對象 )處,拋出未處理的異常:System.ArgumentNullException: 值不能為空!
在代碼段中修改鎖定對象,會出現 blance<0的情況,并會拋出異常
private static readonly object obj = new object();
為什么要設置成只讀的呢?這是因為如果在lock代碼段中改變obj的值,其它線程就暢通無阻了,因為互斥鎖的對象變了,object.ReferenceEquals必然返回false。
所以把上面的修改成private static readonly

來自森大科技官方博客
http://www.cnsendblog.com/index.php/?p=392
GPS平臺、網站建設、軟件開發、系統運維,找森大網絡科技!
http://cnsendnet.taobao.com

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

扶沟县| 兖州市| 庆城县| 荆州市| 铜鼓县| 无为县| 中西区| 雷波县| 漳平市| 安西县| 建阳市| 桂林市| 大同市| 孝昌县| 额济纳旗| 浏阳市| 洪湖市| 凤冈县| 贺州市| 皮山县| 宜川县| 历史| 商洛市| 赤峰市| 镇赉县| 无为县| 新乐市| 贵南县| 永福县| 凤阳县| 祁连县| 蕲春县| 金湖县| 宁晋县| 崇左市| 库伦旗| 龙山县| 逊克县| 宁津县| 望谟县| 客服|