深入多線程之:Reader與Write Locks(讀寫鎖)的使用詳解
線程安全的一個很經(jīng)常的需求是允許并發(fā)讀,但是不允許并發(fā)寫,例如對于文件就是這樣的。
ReaderWriterLockSlim 在.net framework 3.5的時(shí)候就提供了,它是用來代替以前的”fat”版本的”ReaderWriterLock”
這兩個類,有兩種基本的鎖----一個讀鎖,一個寫鎖。
寫鎖是一個完全排他鎖。
讀鎖可以和其他的讀鎖兼容
因此當(dāng)一個線程持有寫鎖的是很,所有的嘗試獲取讀鎖和寫鎖的線程全部阻塞,但是如果沒有一個線程持有寫鎖,那么可以有一系列的線程并發(fā)的獲取讀鎖。
ReaderWriterLockSlim 定義了下面幾個方法來獲取和釋放 讀寫鎖。
Public void EnterReadLock();
Public void ExitReadLock();
Public void EnterWriteLock();
Public void ExitWriteLock();
和Monitor.TryEnter類似,ReaderWriterLockSlim 再對應(yīng)的”EnterXXX”方法上也提供了相應(yīng)的”Try”版本。ReaderWriterLock提供了AcquireXXX 和 ReleaseXXX 方法,當(dāng)超時(shí)發(fā)生了,ReaderWriterLock 拋出一個ApplicationException,而不是返回false。
static readonly ReaderWriterLockSlim _rw = new ReaderWriterLockSlim();
static List<int> _items = new List<int>();
static Random _rand = new Random();
public static void Main()
{
///三個讀線程
new Thread(Read).Start();
new Thread(Read).Start();
new Thread(Read).Start();
//兩個寫線程
new Thread(Write).Start("A");
new Thread(Write).Start("B");
}
static void Read()
{
while (true)
{
_rw.EnterReadLock();//獲取讀鎖
//模擬讀的過程
foreach (int i in _items)
Thread.Sleep(100);
_rw.ExitReadLock();//釋放讀鎖
}
}
static void Write(object threadID)
{
while (true)
{
Console.WriteLine(_rw.CurrentReadCount + " concurrent readers");
int newNumber = GetRandomNum(100);
_rw.EnterWriteLock(); //獲取寫鎖
_items.Add(newNumber); //寫數(shù)據(jù)
_rw.ExitWriteLock(); //釋放寫鎖
Console.WriteLine("Thread " + threadID + " added " + newNumber);
Thread.Sleep(100);
}
}
//獲取隨機(jī)數(shù)
static int GetRandomNum(int max) { lock (_rand) return _rand.Next(max); }
再實(shí)際的發(fā)布版本中,最好使用try/finally 來確保即使異常拋出了,鎖也被正確的釋放了。
像CurrentReadCount 屬性,ReaderWriterLockSlim 提供了以下屬性用來監(jiān)視鎖。
可更新鎖:
再一個原子操作里將讀鎖升級為寫鎖是很有用的,例如,假設(shè)你想要再一個list 里面寫一些不存在的項(xiàng)的時(shí)候, 你可能會執(zhí)行下面的一些步驟:
問題是:在第三步和第四步之間,可能有另一個線程修改了列表。
ReaderWriterLockSlim 通過一個叫做可更新鎖( upgradeable lock),來解決這個問題。
一個可更新鎖除了它可以在一個原子操作中變成寫鎖外很像一個讀鎖,你可以這樣使用它:
- 調(diào)用EnterUpgradeableReadLock 獲取可更新鎖。執(zhí)行一些讀操作,例如判斷要寫的東西在不在List中。調(diào)用EnterWriteLock , 這個方法會將可更新鎖 升級為 寫鎖。執(zhí)行寫操作,調(diào)用ExitWriteLock 方法,這個方法將寫鎖轉(zhuǎn)換回可更新鎖。繼續(xù)執(zhí)行一些讀操作,或什么都不做。
從調(diào)用者的角度來看,它很像一個嵌套/遞歸鎖,從功能上講,在第三步,
ReaderWriterLockSlim 在一個原子操作里面釋放讀鎖,然后獲取寫鎖。
可更新鎖和讀鎖的重要區(qū)別是:盡管可更新鎖可以和讀鎖共存,但是一次只能有一個可更新鎖被獲取。這樣的主要目的是防止死鎖。
這樣我們可以修改Write方法,讓它可以添加一些不在列表中的Item。 int newNumber = GetRandomNum(100); _rw.EnterUpgradeableReadLock(); //獲取可更新鎖 Thread.Sleep(100);
static void Write(object threadID)
{
while (true)
{
Console.WriteLine(_rw.CurrentReadCount + " concurrent readers");
if (!_items.Contains(newNumber)) //如果要寫的東西不在列表中
{
_rw.EnterWriteLock(); //可更新鎖變成寫鎖
_items.Add(newNumber); //寫東西
_rw.ExitWriteLock(); //重新變回可更新鎖
Console.WriteLine("Thread " + threadID + " added " + newNumber); //讀數(shù)據(jù)
}
_rw.ExitUpgradeableReadLock(); //退出可更新鎖
}
}
從上面的例子可以看到C#提供的讀寫鎖功能強(qiáng)大,使用方便,
所以在自己編寫讀寫鎖的時(shí)候,要考慮下是否需要支持可更新鎖,是否有必要自己寫一個讀寫鎖.
相關(guān)文章
C#使用分部類設(shè)計(jì)實(shí)現(xiàn)一個計(jì)算器
分部類是C#4.5中的一個新特性,它的出現(xiàn)使得程序的結(jié)構(gòu)更加合理,代碼組織更加緊密,本文將使用分部類設(shè)計(jì)實(shí)現(xiàn)一個簡單的計(jì)算器,感興趣的小伙伴可以了解下2024-02-02C#入門學(xué)習(xí)之集合、比較和轉(zhuǎn)換
本文詳細(xì)講解了C#中的集合、比較和轉(zhuǎn)換,文中通過示例代碼介紹的非常詳細(xì)。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-05-05詳解C#使用AD(Active Directory)驗(yàn)證內(nèi)網(wǎng)用戶名密碼
這篇文章主要介紹了詳解C#使用AD(Active Directory)驗(yàn)證內(nèi)網(wǎng)用戶名密碼的相關(guān)資料,希望通過本文能幫助到大家,讓大家實(shí)現(xiàn)這樣的功能,需要的朋友可以參考下2017-10-10