深入多線程之:Reader與Write Locks(讀寫鎖)的使用詳解
線程安全的一個很經(jīng)常的需求是允許并發(fā)讀,但是不允許并發(fā)寫,例如對于文件就是這樣的。
ReaderWriterLockSlim 在.net framework 3.5的時候就提供了,它是用來代替以前的”fat”版本的”ReaderWriterLock”
這兩個類,有兩種基本的鎖----一個讀鎖,一個寫鎖。
寫鎖是一個完全排他鎖。
讀鎖可以和其他的讀鎖兼容
因此當一個線程持有寫鎖的是很,所有的嘗試獲取讀鎖和寫鎖的線程全部阻塞,但是如果沒有一個線程持有寫鎖,那么可以有一系列的線程并發(fā)的獲取讀鎖。
ReaderWriterLockSlim 定義了下面幾個方法來獲取和釋放 讀寫鎖。
Public void EnterReadLock();
Public void ExitReadLock();
Public void EnterWriteLock();
Public void ExitWriteLock();
和Monitor.TryEnter類似,ReaderWriterLockSlim 再對應的”EnterXXX”方法上也提供了相應的”Try”版本。ReaderWriterLock提供了AcquireXXX 和 ReleaseXXX 方法,當超時發(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);
}
}
//獲取隨機數(shù)
static int GetRandomNum(int max) { lock (_rand) return _rand.Next(max); }
再實際的發(fā)布版本中,最好使用try/finally 來確保即使異常拋出了,鎖也被正確的釋放了。
像CurrentReadCount 屬性,ReaderWriterLockSlim 提供了以下屬性用來監(jiān)視鎖。
可更新鎖:
再一個原子操作里將讀鎖升級為寫鎖是很有用的,例如,假設你想要再一個list 里面寫一些不存在的項的時候, 你可能會執(zhí)行下面的一些步驟:
問題是:在第三步和第四步之間,可能有另一個線程修改了列表。
ReaderWriterLockSlim 通過一個叫做可更新鎖( upgradeable lock),來解決這個問題。
一個可更新鎖除了它可以在一個原子操作中變成寫鎖外很像一個讀鎖,你可以這樣使用它:
- 調用EnterUpgradeableReadLock 獲取可更新鎖。執(zhí)行一些讀操作,例如判斷要寫的東西在不在List中。調用EnterWriteLock , 這個方法會將可更新鎖 升級為 寫鎖。執(zhí)行寫操作,調用ExitWriteLock 方法,這個方法將寫鎖轉換回可更新鎖。繼續(xù)執(zhí)行一些讀操作,或什么都不做。
從調用者的角度來看,它很像一個嵌套/遞歸鎖,從功能上講,在第三步,
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#提供的讀寫鎖功能強大,使用方便,
所以在自己編寫讀寫鎖的時候,要考慮下是否需要支持可更新鎖,是否有必要自己寫一個讀寫鎖.
相關文章
詳解C#使用AD(Active Directory)驗證內網(wǎng)用戶名密碼
這篇文章主要介紹了詳解C#使用AD(Active Directory)驗證內網(wǎng)用戶名密碼的相關資料,希望通過本文能幫助到大家,讓大家實現(xiàn)這樣的功能,需要的朋友可以參考下2017-10-10