來自森大科技官方博客
http://www.cnsendblog.com/index.php/?p=392
GPS平臺、網站建設、軟件開發(fā)、系統(tǒng)運維,找森大網絡科技!
http://cnsendnet.taobao.com
“專業(yè)、務實、高效、創(chuàng)新、把客戶的事當成自己的事”是我們每一個人一直以來堅持追求的企業(yè)文化。 成都創(chuàng)新互聯(lián)是您可以信賴的網站建設服務商、專業(yè)的互聯(lián)網服務提供商! 專注于成都網站設計、做網站、軟件開發(fā)、設計服務業(yè)務。我們始終堅持以客戶需求為導向,結合用戶體驗與視覺傳達,提供有針對性的項目解決方案,提供專業(yè)性的建議,創(chuàng)新互聯(lián)建站將不斷地超越自我,追逐市場,引領市場!
一. 為什么要lock,lock了什么?
當我們使用線程的時候,效率最高的方式當然是異步,即各個線程同時運行,其間不相互依賴和等待。但當不同的線程都需要訪問某個資源的時候,就需要同步機制了,也就是說當對同一個資源進行讀寫的時候,我們要使該資源在同一時刻只能被一個線程操作,以確保每個操作都是有效即時的,也即保證其操作的原子性。lock是C#中最常用的同步方式,格式為lock(objectA){codeB} 。
lock(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);
//在主線程中l(wèi)ock c1
lock (c1)
{
//調用沒有被lock的方法
c1.DoNotLockMe();
//調用被lock的方法,并試圖將deadlock解除
c1.LockMe(false);
}
}
}
在t1線程中,LockMe調用了lock(this), 也就是Main函數中的c1,這時候在主線程中調用lock(c1)時,必須要等待t1中的lock塊執(zhí)行完畢之后才能訪問c1,即所有c1相關的操作都無法完成,于是我們看到連c1.DoNotLockMe()都沒有執(zhí)行。
把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中僅僅鎖定這個私有l(wèi)ocker,而不是整個對象。這時候重新運行程序,可以看到雖然t1出現了死鎖,DoNotLockMe()仍然可以由主線程訪問;LockMe()依然不能訪問,原因是其中鎖定的locker還沒有被t1釋放。
關鍵點:
kenny add
而對于Monitor,發(fā)現它的靜態(tài)方法Enter(object obj)有一個異常類型ArgumentNullException,
執(zhí)行l(wèi)ock(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平臺、網站建設、軟件開發(fā)、系統(tǒng)運維,找森大網絡科技!
http://cnsendnet.taobao.com
文章名稱:為什么不要lock(this),鎖定變量最好是readonly
分享地址:http://aaarwkj.com/article34/pccdse.html
成都網站建設公司_創(chuàng)新互聯(lián),為您提供ChatGPT、網站營銷、服務器托管、小程序開發(fā)、網站設計、域名注冊
聲明:本網站發(fā)布的內容(圖片、視頻和文字)以用戶投稿、用戶轉載內容為主,如果涉及侵權請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內容未經允許不得轉載,或轉載時需注明來源: 創(chuàng)新互聯(lián)