国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > C# > 正文

如何使用C#讀寫鎖ReaderWriterLockSlim

2019-10-29 21:40:06
字體:
來源:轉載
供稿:網友

這篇文章向大家介紹了讀寫鎖ReaderWriterLockSlim,其優點就是多個線程可以同時讀取該對象,要了解更多讀寫鎖的知識,仔細閱讀下文吧

讀寫鎖的概念很簡單,允許多個線程同時獲取讀鎖,但同一時間只允許一個線程獲得寫鎖,因此也稱作共享-獨占鎖。在C#中,推薦使用ReaderWriterLockSlim類來完成讀寫鎖的功能。

某些場合下,對一個對象的讀取次數遠遠大于修改次數,如果只是簡單的用lock方式加鎖,則會影響讀取的效率。而如果采用讀寫鎖,則多個線程可以同時讀取該對象,只有等到對象被寫入鎖占用的時候,才會阻塞。

簡單的說,當某個線程進入讀取模式時,此時其他線程依然能進入讀取模式,假設此時一個線程要進入寫入模式,那么他不得不被阻塞。直到讀取模式退出為止。

同樣的,如果某個線程進入了寫入模式,那么其他線程無論是要寫入還是讀取,都是會被阻塞的。

進入寫入/讀取模式有2種方法:

EnterReadLock嘗試進入寫入模式鎖定狀態。

TryEnterReadLock(Int32) 嘗試進入讀取模式鎖定狀態,可以選擇整數超時時間。

EnterWriteLock 嘗試進入寫入模式鎖定狀態。

TryEnterWriteLock(Int32) 嘗試進入寫入模式鎖定狀態,可以選擇超時時間。

退出寫入/讀取模式有2種方法:

ExitReadLock 減少讀取模式的遞歸計數,并在生成的計數為 0(零)時退出讀取模式。

ExitWriteLock 減少寫入模式的遞歸計數,并在生成的計數為 0(零)時退出寫入模式。

下面演示一下用法:

 

 
  1. public class Program 
  2. static private ReaderWriterLockSlim rwl = new ReaderWriterLockSlim(); 
  3. static void Main(string[] args) 
  4. Thread t_read1 = new Thread(new ThreadStart(ReadSomething)); 
  5. t_read1.Start(); 
  6. Console.WriteLine("{0} Create Thread ID {1} , Start ReadSomething", DateTime.Now.ToString("hh:mm:ss fff"), t_read1.GetHashCode()); 
  7. Thread t_read2 = new Thread(new ThreadStart(ReadSomething)); 
  8. t_read2.Start(); 
  9. Console.WriteLine("{0} Create Thread ID {1} , Start ReadSomething", DateTime.Now.ToString("hh:mm:ss fff"), t_read2.GetHashCode()); 
  10. Thread t_write1 = new Thread(new ThreadStart(WriteSomething)); 
  11. t_write1.Start(); 
  12. Console.WriteLine("{0} Create Thread ID {1} , Start WriteSomething", DateTime.Now.ToString("hh:mm:ss fff"), t_write1.GetHashCode()); 
  13. static public void ReadSomething() 
  14. Console.WriteLine("{0} Thread ID {1} Begin EnterReadLock...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode()); 
  15. rwl.EnterReadLock(); 
  16. try 
  17. Console.WriteLine("{0} Thread ID {1} reading sth...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode()); 
  18. Thread.Sleep(5000);//模擬讀取信息 
  19. Console.WriteLine("{0} Thread ID {1} reading end.", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode()); 
  20. finally 
  21. rwl.ExitReadLock(); 
  22. Console.WriteLine("{0} Thread ID {1} ExitReadLock...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode()); 
  23. static public void WriteSomething() 
  24. Console.WriteLine("{0} Thread ID {1} Begin EnterWriteLock...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode()); 
  25. rwl.EnterWriteLock(); 
  26. try 
  27. Console.WriteLine("{0} Thread ID {1} writing sth...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode()); 
  28. Thread.Sleep(10000);//模擬寫入信息 
  29. Console.WriteLine("{0} Thread ID {1} writing end.", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode()); 
  30. finally 
  31. rwl.ExitWriteLock(); 
  32. Console.WriteLine("{0} Thread ID {1} ExitWriteLock...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode()); 

如何使用C#讀寫鎖ReaderWriterLockSlim

可以看到3號線程和4號線程能夠同時進入讀模式,而5號線程過了5秒鐘后(即3,4號線程退出讀鎖后),才能進入寫模式。

把上述代碼修改一下,先開啟2個寫模式的線程,然后在開啟讀模式線程,代碼如下:

 

 
  1. static void Main(string[] args) 
  2. Thread t_write1 = new Thread(new ThreadStart(WriteSomething)); 
  3. t_write1.Start(); 
  4. Console.WriteLine("{0} Create Thread ID {1} , Start WriteSomething", DateTime.Now.ToString("hh:mm:ss fff"), t_write1.GetHashCode()); 
  5. Thread t_write2 = new Thread(new ThreadStart(WriteSomething)); 
  6. t_write2.Start(); 
  7. Console.WriteLine("{0} Create Thread ID {1} , Start WriteSomething", DateTime.Now.ToString("hh:mm:ss fff"), t_write2.GetHashCode()); 
  8. Thread t_read1 = new Thread(new ThreadStart(ReadSomething)); 
  9. t_read1.Start(); 
  10. Console.WriteLine("{0} Create Thread ID {1} , Start ReadSomething", DateTime.Now.ToString("hh:mm:ss fff"), t_read1.GetHashCode()); 
  11. Thread t_read2 = new Thread(new ThreadStart(ReadSomething)); 
  12. t_read2.Start(); 
  13. Console.WriteLine("{0} Create Thread ID {1} , Start ReadSomething", DateTime.Now.ToString("hh:mm:ss fff"), t_read2.GetHashCode()); 

結果如下:

如何使用C#讀寫鎖ReaderWriterLockSlim

可以看到,3號線程和4號線程都要進入寫模式,但是3號線程先占用寫入鎖,因此4號線程不得不等了10s后才進入。5號線程和6號線程需要占用讀取鎖,因此等4號線程退出寫入鎖后才能繼續下去。

TryEnterReadLock和TryEnterWriteLock可以設置一個超時時間,運行到這句話的時候,線程會阻塞在此,如果此時能占用鎖,那么返回true,如果到超時時間還未占用鎖,那么返回false,放棄鎖的占用,直接繼續執行下面的代碼。

EnterUpgradeableReadLock

ReaderWriterLockSlim類提供了可升級讀模式,這種方式和讀模式的區別在于它還有通過調用 EnterWriteLock 或 TryEnterWriteLock 方法升級為寫入模式。 因為每次只能有一個線程處于可升級模式。進入可升級模式的線程,不會影響讀取模式的線程,即當一個線程進入可升級模式,任意數量線程可以同時進入讀取模式,不會阻塞。如果有多個線程已經在等待獲取寫入鎖,那么運行EnterUpgradeableReadLock將會阻塞,直到那些線程超時或者退出寫入鎖。

下面代碼演示了如何在可升級讀模式下,升級到寫入鎖。

 

 
  1. static public void UpgradeableRead() 
  2. Console.WriteLine("{0} Thread ID {1} Begin EnterUpgradeableReadLock...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode()); 
  3. rwl.EnterUpgradeableReadLock(); 
  4. try 
  5. Console.WriteLine("{0} Thread ID {1} doing sth...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode()); 
  6. Console.WriteLine("{0} Thread ID {1} Begin EnterWriteLock...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode()); 
  7. rwl.EnterWriteLock(); 
  8. try 
  9. Console.WriteLine("{0} Thread ID {1} writing sth...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode()); 
  10. Thread.Sleep(10000);//模擬寫入信息 
  11. Console.WriteLine("{0} Thread ID {1} writing end.", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode()); 
  12. finally 
  13. rwl.ExitWriteLock(); 
  14. Console.WriteLine("{0} Thread ID {1} ExitWriteLock...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode()); 
  15. Thread.Sleep(10000);//模擬讀取信息 
  16. Console.WriteLine("{0} Thread ID {1} doing end.", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode()); 
  17. finally 
  18. rwl.ExitUpgradeableReadLock(); 
  19. Console.WriteLine("{0} Thread ID {1} ExitUpgradeableReadLock...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode()); 

讀寫鎖對于性能的影響是明顯的。

下面測試代碼:

 

 
  1. public class Program 
  2. static private ReaderWriterLockSlim rwl = new ReaderWriterLockSlim(); 
  3. static void Main(string[] args) 
  4. Stopwatch sw = new Stopwatch(); 
  5. sw.Start(); 
  6. List<Task> lstTask = new List<Task>(); 
  7. for (int i = 0; i < 500; i++) 
  8. if (i % 25 != 0) 
  9. var t = Task.Factory.StartNew(ReadSomething); 
  10. lstTask.Add(t); 
  11. else 
  12. var t = Task.Factory.StartNew(WriteSomething); 
  13. lstTask.Add(t); 
  14. Task.WaitAll(lstTask.ToArray()); 
  15. sw.Stop(); 
  16. Console.WriteLine("使用ReaderWriterLockSlim方式,耗時:" + sw.Elapsed); 
  17. sw.Restart(); 
  18. lstTask = new List<Task>(); 
  19. for (int i = 0; i < 500; i++) 
  20. if (i % 25 != 0) 
  21. var t = Task.Factory.StartNew(ReadSomething_lock); 
  22. lstTask.Add(t); 
  23. else 
  24. var t = Task.Factory.StartNew(WriteSomething_lock); 
  25. lstTask.Add(t); 
  26. Task.WaitAll(lstTask.ToArray()); 
  27. sw.Stop(); 
  28. Console.WriteLine("使用lock方式,耗時:" + sw.Elapsed); 
  29. static private object _lock1 = new object(); 
  30. static public void ReadSomething_lock() 
  31. lock (_lock1) 
  32. //Console.WriteLine("{0} Thread ID {1} reading sth...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode()); 
  33. Thread.Sleep(10);//模擬讀取信息 
  34. //Console.WriteLine("{0} Thread ID {1} reading end.", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode()); 
  35. static public void WriteSomething_lock() 
  36. lock (_lock1) 
  37. //Console.WriteLine("{0} Thread ID {1} writing sth...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode()); 
  38. Thread.Sleep(100);//模擬寫入信息 
  39. //Console.WriteLine("{0} Thread ID {1} writing end.", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode()); 
  40. static public void ReadSomething() 
  41. rwl.EnterReadLock(); 
  42. try 
  43. //Console.WriteLine("{0} Thread ID {1} reading sth...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode()); 
  44. Thread.Sleep(10);//模擬讀取信息 
  45. //Console.WriteLine("{0} Thread ID {1} reading end.", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode()); 
  46. finally 
  47. rwl.ExitReadLock(); 
  48. static public void WriteSomething() 
  49. rwl.EnterWriteLock(); 
  50. try 
  51. //Console.WriteLine("{0} Thread ID {1} writing sth...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode()); 
  52. Thread.Sleep(100);//模擬寫入信息 
  53. //Console.WriteLine("{0} Thread ID {1} writing end.", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode()); 
  54. finally 
  55. rwl.ExitWriteLock(); 

上述代碼,就500個Task,每個Task占用一個線程池線程,其中20個寫入線程和480個讀取線程,模擬操作。其中讀取數據花10ms,寫入操作花100ms,分別測試了對于lock方式和ReaderWriterLockSlim方式??梢宰鲆粋€估算,對于ReaderWriterLockSlim,假設480個線程同時讀取,那么消耗10ms,20個寫入操作占用2000ms,因此所消耗時間2010ms,而對于普通的lock方式,由于都是獨占性的,因此480個讀取操作占時間4800ms+20個寫入操作2000ms=6800ms。運行結果顯示了性能提升明顯。

如何使用C#讀寫鎖ReaderWriterLockSlim

以上是本文的全部內容,希望對大家熟練應用讀寫鎖ReaderWriterLockSlim有所幫助。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 驻马店市| 郎溪县| 佳木斯市| 成武县| 应城市| 常宁市| 云南省| 汉中市| 资中县| 西安市| 时尚| 望奎县| 永兴县| 肥城市| 龙海市| 望江县| 广饶县| 乌海市| 安康市| 胶南市| 黎城县| 蕲春县| 麻栗坡县| 郓城县| 闽清县| 浦东新区| 张掖市| 陈巴尔虎旗| 任丘市| 丰城市| 万年县| 太谷县| 山阴县| 米易县| 南昌市| 五家渠市| 临清市| 永福县| 平安县| 漾濞| 赞皇县|