一. 為什么要lock,lock了什么?
成都創(chuàng)新互聯(lián)成立與2013年,先為連江等服務(wù)建站,連江等地企業(yè),進(jìn)行企業(yè)商務(wù)咨詢服務(wù)。為連江企業(yè)網(wǎng)站制作PC+手機(jī)+微官網(wǎng)三網(wǎng)同步一站式服務(wù)解決您的所有建站問題。
當(dāng)我們使用線程的時(shí)候,效率最高的方式當(dāng)然是異步,即各個(gè)線程同時(shí)運(yùn)行,其間不相互依賴和等待。但當(dāng)不同的線程都需要訪問某個(gè)資源的時(shí)候,就需要同步機(jī)制了,也就是說當(dāng)對(duì)同一個(gè)資源進(jìn)行讀寫的時(shí)候,我們要使該資源在同一時(shí)刻只能被一個(gè)線程操作,以確保每個(gè)操作都是有效即時(shí)的,也即保證其操作的原子性。lock是C#中最常用的同步方式,格式為lock(objectA){codeB} 。
lock(objectA){codeB} 看似簡單,實(shí)際上有三個(gè)意思,這對(duì)于適當(dāng)?shù)厥褂盟陵P(guān)重要:
二. lock(this)示例
我們看一個(gè)例子:
using System;
using System.Threading;
namespace Namespace1
{
class C1
{
private bool deadlocked= true;
//這個(gè)方法用到了lock,我們希望lock的代碼在同一時(shí)刻只能由一個(gè)線程訪問
public void LockMe(object o)
{
lock (this)
{
while(deadlocked)
{
deadlocked = (bool)o;
Console.WriteLine("Foo: I am locked :(");
Thread.Sleep(500);
}
}
}
//所有線程都可以同時(shí)訪問的方法
public void DoNotLockMe()
{
Console.WriteLine("I am not locked :)");
}
}
class Program
{
staticvoid Main(string[] args)
{
C1 c1 =new C1();
//在t1線程中調(diào)用LockMe,并將deadlock設(shè)為true(將出現(xiàn)死鎖)
Thread t1= new Thread(c1.LockMe);
t1.Start(true);
Thread.Sleep(100);
//在主線程中l(wèi)ock c1
lock (c1)
{
//調(diào)用沒有被lock的方法
c1.DoNotLockMe();
//調(diào)用被lock的方法,并試圖將deadlock解除
c1.LockMe(false);
}
}
}
在t1線程中,LockMe調(diào)用了lock(this), 也就是Main函數(shù)中的c1,這時(shí)候在主線程中調(diào)用lock(c1)時(shí),必須要等待t1中的lock塊執(zhí)行完畢之后才能訪問c1,即所有c1相關(guān)的操作都無法完成,于是我們看到連c1.DoNotLockMe()都沒有執(zhí)行。
把C1的代碼稍作改動(dòng):
class C1
{
privatebool deadlocked= true;
private object locker= new object();
//這個(gè)方法用到了lock,我們希望lock的代碼在同一時(shí)刻只能由一個(gè)線程訪問
public void LockMe(object o)
{
lock (locker)
{
while(deadlocked)
{
deadlocked = (bool)o;
Console.WriteLine("Foo: I am locked :(");
Thread.Sleep(500);
}
}
}
//所有線程都可以同時(shí)訪問的方法
public void DoNotLockMe()
{
Console.WriteLine("I am not locked :)");
}
}
這次我們使用一個(gè)私有成員作為鎖定變量(locker),在LockMe中僅僅鎖定這個(gè)私有l(wèi)ocker,而不是整個(gè)對(duì)象。這時(shí)候重新運(yùn)行程序,可以看到雖然t1出現(xiàn)了死鎖,DoNotLockMe()仍然可以由主線程訪問;LockMe()依然不能訪問,原因是其中鎖定的locker還沒有被t1釋放。
關(guān)鍵點(diǎn):
kenny add
而對(duì)于Monitor,發(fā)現(xiàn)它的靜態(tài)方法Enter(object obj)有一個(gè)異常類型ArgumentNullException,
執(zhí)行l(wèi)ock(null對(duì)象 )處,拋出未處理的異常:System.ArgumentNullException: 值不能為空!
在代碼段中修改鎖定對(duì)象,會(huì)出現(xiàn) blance<0的情況,并會(huì)拋出異常
private static readonly object obj = new object();
為什么要設(shè)置成只讀的呢?這是因?yàn)槿绻趌ock代碼段中改變obj的值,其它線程就暢通無阻了,因?yàn)榛コ怄i的對(duì)象變了,object.ReferenceEquals必然返回false
分享標(biāo)題:C#中l(wèi)ock的使用方法
當(dāng)前URL:http://jinyejixie.com/article16/psiodg.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供網(wǎng)站內(nèi)鏈、小程序開發(fā)、App設(shè)計(jì)、域名注冊(cè)、虛擬主機(jī)、網(wǎng)站制作
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場,如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來源: 創(chuàng)新互聯(lián)