ReaderWriterLock 類別

定義

定義鎖定,它支援單一寫入器和多重讀取器。

public sealed class ReaderWriterLock : System.Runtime.ConstrainedExecution.CriticalFinalizerObject
public sealed class ReaderWriterLock
[System.Runtime.InteropServices.ComVisible(true)]
public sealed class ReaderWriterLock : System.Runtime.ConstrainedExecution.CriticalFinalizerObject
繼承
ReaderWriterLock
繼承
ReaderWriterLock
屬性

範例

下列範例示範如何使用 ReaderWriterLock 來保護共用資源、名為 resource 的整數值,由多個執行緒同時讀取和寫入。 請注意,會在類別層級宣告 , ReaderWriterLock 讓所有線程都能看見它。

// The complete code is located in the ReaderWriterLock class topic.
using System;
using System.Threading;

public class Example
{
   static ReaderWriterLock rwl = new ReaderWriterLock();
   // Define the shared resource protected by the ReaderWriterLock.
   static int resource = 0;

   const int numThreads = 26;
   static bool running = true;

   // Statistics.
   static int readerTimeouts = 0;
   static int writerTimeouts = 0;
   static int reads = 0;
   static int writes = 0;

   public static void Main()
   {
      // Start a series of threads to randomly read from and
      // write to the shared resource.
      Thread[] t = new Thread[numThreads];
      for (int i = 0; i < numThreads; i++){
         t[i] = new Thread(new ThreadStart(ThreadProc));
         t[i].Name = new String((char)(i + 65), 1);
         t[i].Start();
         if (i > 10)
            Thread.Sleep(300);
      }

      // Tell the threads to shut down and wait until they all finish.
      running = false;
      for (int i = 0; i < numThreads; i++)
         t[i].Join();

      // Display statistics.
      Console.WriteLine("\n{0} reads, {1} writes, {2} reader time-outs, {3} writer time-outs.",
            reads, writes, readerTimeouts, writerTimeouts);
      Console.Write("Press ENTER to exit... ");
      Console.ReadLine();
   }

   static void ThreadProc()
   {
      Random rnd = new Random();

      // Randomly select a way for the thread to read and write from the shared
      // resource.
      while (running) {
         double action = rnd.NextDouble();
         if (action < .8)
            ReadFromResource(10);
         else if (action < .81)
            ReleaseRestore(rnd, 50);
         else if (action < .90)
            UpgradeDowngrade(rnd, 100);
         else
            WriteToResource(rnd, 100);
      }
   }

   // Request and release a reader lock, and handle time-outs.
   static void ReadFromResource(int timeOut)
   {
      try {
         rwl.AcquireReaderLock(timeOut);
         try {
            // It is safe for this thread to read from the shared resource.
            Display("reads resource value " + resource);
            Interlocked.Increment(ref reads);
         }
         finally {
            // Ensure that the lock is released.
            rwl.ReleaseReaderLock();
         }
      }
      catch (ApplicationException) {
         // The reader lock request timed out.
         Interlocked.Increment(ref readerTimeouts);
      }
   }

   // Request and release the writer lock, and handle time-outs.
   static void WriteToResource(Random rnd, int timeOut)
   {
      try {
         rwl.AcquireWriterLock(timeOut);
         try {
            // It's safe for this thread to access from the shared resource.
            resource = rnd.Next(500);
            Display("writes resource value " + resource);
            Interlocked.Increment(ref writes);
         }
         finally {
            // Ensure that the lock is released.
            rwl.ReleaseWriterLock();
         }
      }
      catch (ApplicationException) {
         // The writer lock request timed out.
         Interlocked.Increment(ref writerTimeouts);
      }
   }

   // Requests a reader lock, upgrades the reader lock to the writer
   // lock, and downgrades it to a reader lock again.
   static void UpgradeDowngrade(Random rnd, int timeOut)
   {
      try {
         rwl.AcquireReaderLock(timeOut);
         try {
            // It's safe for this thread to read from the shared resource.
            Display("reads resource value " + resource);
            Interlocked.Increment(ref reads);

            // To write to the resource, either release the reader lock and
            // request the writer lock, or upgrade the reader lock. Upgrading
            // the reader lock puts the thread in the write queue, behind any
            // other threads that might be waiting for the writer lock.
            try {
               LockCookie lc = rwl.UpgradeToWriterLock(timeOut);
               try {
                  // It's safe for this thread to read or write from the shared resource.
                  resource = rnd.Next(500);
                  Display("writes resource value " + resource);
                  Interlocked.Increment(ref writes);
               }
               finally {
                  // Ensure that the lock is released.
                  rwl.DowngradeFromWriterLock(ref lc);
               }
            }
            catch (ApplicationException) {
               // The upgrade request timed out.
               Interlocked.Increment(ref writerTimeouts);
            }

            // If the lock was downgraded, it's still safe to read from the resource.
            Display("reads resource value " + resource);
            Interlocked.Increment(ref reads);
         }
         finally {
            // Ensure that the lock is released.
            rwl.ReleaseReaderLock();
         }
      }
      catch (ApplicationException) {
         // The reader lock request timed out.
         Interlocked.Increment(ref readerTimeouts);
      }
   }

   // Release all locks and later restores the lock state.
   // Uses sequence numbers to determine whether another thread has
   // obtained a writer lock since this thread last accessed the resource.
   static void ReleaseRestore(Random rnd, int timeOut)
   {
      int lastWriter;

      try {
         rwl.AcquireReaderLock(timeOut);
         try {
            // It's safe for this thread to read from the shared resource,
            // so read and cache the resource value.
            int resourceValue = resource;     // Cache the resource value.
            Display("reads resource value " + resourceValue);
            Interlocked.Increment(ref reads);

            // Save the current writer sequence number.
            lastWriter = rwl.WriterSeqNum;

            // Release the lock and save a cookie so the lock can be restored later.
            LockCookie lc = rwl.ReleaseLock();

            // Wait for a random interval and then restore the previous state of the lock.
            Thread.Sleep(rnd.Next(250));
            rwl.RestoreLock(ref lc);

            // Check whether other threads obtained the writer lock in the interval.
            // If not, then the cached value of the resource is still valid.
            if (rwl.AnyWritersSince(lastWriter)) {
               resourceValue = resource;
               Interlocked.Increment(ref reads);
               Display("resource has changed " + resourceValue);
            }
            else {
               Display("resource has not changed " + resourceValue);
            }
         }
         finally {
            // Ensure that the lock is released.
            rwl.ReleaseReaderLock();
         }
      }
      catch (ApplicationException) {
         // The reader lock request timed out.
         Interlocked.Increment(ref readerTimeouts);
      }
   }

   // Helper method briefly displays the most recent thread action.
   static void Display(string msg)
   {
      Console.Write("Thread {0} {1}.       \r", Thread.CurrentThread.Name, msg);
   }
}

備註

重要

.NET Framework有兩個讀取器寫入器鎖定和 ReaderWriterLockSlimReaderWriterLock 。 建議針對所有新的開發使用 ReaderWriterLockSlimReaderWriterLockSlim 類似於 ReaderWriterLock,但是它有遞迴以及升級和降級鎖定狀態的簡化規則。 ReaderWriterLockSlim 可避免可能發生死結的許多情況。 此外,ReaderWriterLockSlim 的效能明顯優於 ReaderWriterLock

ReaderWriterLock 用來同步處理資源的存取。 在任何指定時間,它都會允許多個執行緒的並行讀取權限,或允許單一線程的寫入權限。 在資源不常變更的情況下,提供 ReaderWriterLock 比簡單的一次性鎖定更好的輸送量,例如 Monitor

ReaderWriterLock 最適合在讀取大部分的存取位置,而寫入不常且持續時間較短。 多個讀取器會替代單一寫入器,讓讀取器或寫入器不會長時間遭到封鎖。

注意

長時間保留讀取器鎖定或寫入器鎖定將會耗盡其他執行緒。 為了獲得最佳效能,請考慮重建您的應用程式,以將寫入持續時間降到最低。

執行緒可以保存讀取器鎖定或寫入器鎖定,但不能同時保存兩者。 您可以使用 UpgradeToWriterLockDowngradeFromWriterLock ,而不是釋放讀取器鎖定來取得寫入器鎖定。

遞迴鎖定要求會增加鎖定的鎖定計數。

讀取器和寫入器會個別排入佇列。 當執行緒釋放寫入器鎖定時,該瞬間在讀取器佇列中等候的所有線程都會被授與讀取器鎖定;當所有讀取器鎖定都已釋放時,在寫入器佇列中等候的下一個執行緒,如果有的話,就會被授與寫入器鎖定等等。 換句話說, ReaderWriterLock 讀取器集合與一個寫入器之間的替代。

雖然寫入器佇列中的執行緒正在等候釋放作用中的讀取器鎖定,但要求新讀取器鎖定的執行緒會累積在讀取器佇列中。 即使其可以與現有的讀取器鎖定持有者共用平行存取權,也不會授與其要求;這有助於防止讀取器無限期封鎖寫入器。

取得接受逾時值鎖定 ReaderWriterLock 的大部分方法。 使用逾時來避免應用程式中的死結。 例如,執行緒可能會取得一個資源的寫入器鎖定,然後在第二個資源上要求讀取器鎖定;同時,另一個執行緒可能會取得第二個資源的寫入器鎖定,並在第一個資源上要求讀取器鎖定。 除非使用逾時,否則執行緒會死結。

如果逾時間隔過期,而且尚未授與鎖定要求,方法會擲 ApplicationException 回 ,以將控制權傳回給呼叫執行緒。 執行緒可以攔截此例外狀況,並判斷接下來要採取的動作。

逾時以毫碼錶示。 如果您使用 System.TimeSpan 來指定逾時,所使用的值就是 所 TimeSpan 表示的完整毫秒總數。 下表顯示以毫秒為單位的有效逾時值。

描述
-1 執行緒會等到取得鎖定為止,而不論花費多少時間。 對於指定整數逾時的方法,可以使用常數 Infinite
0 執行緒不會等候取得鎖定。 如果無法立即取得鎖定,方法會傳回 。
>0 要等候的毫秒數。

除了 -1 之外,不允許負逾時值。 如果您指定 -1 以外的負整數,則會改用零的逾時值。 (也就是說,如果無法立即取得鎖定,則方法會傳回 ,) 如果您指定 TimeSpan 代表 -1 以外的負數毫秒數的 , ArgumentOutOfRangeException 則會擲回。

建構函式

ReaderWriterLock()

初始化 ReaderWriterLock 類別的新執行個體。

屬性

IsReaderLockHeld

取得值,指出目前的執行緒是否掌握讀取器的鎖定。

IsWriterLockHeld

取得值,指出目前的執行緒是否掌握寫入器的鎖定。

WriterSeqNum

取得目前的序號。

方法

AcquireReaderLock(Int32)

使用逾時值 Int32 取得讀取器的鎖定。

AcquireReaderLock(TimeSpan)

使用逾時值 TimeSpan 取得讀取器的鎖定。

AcquireWriterLock(Int32)

使用逾時值 Int32 取得寫入器的鎖定。

AcquireWriterLock(TimeSpan)

使用逾時值 TimeSpan 取得寫入器的鎖定。

AnyWritersSince(Int32)

指示取得序號之後有無將寫入器鎖定授與至任何執行緒。

DowngradeFromWriterLock(LockCookie)

將執行緒的鎖定狀態還原到呼叫 UpgradeToWriterLock(Int32) 之前的狀態。

Equals(Object)

判斷指定的物件是否等於目前的物件。

(繼承來源 Object)
Finalize()

確認釋出資源,並在記憶體回收行程回收 ReaderWriterLock 物件時執行其他清除作業。

GetHashCode()

做為預設雜湊函式。

(繼承來源 Object)
GetType()

取得目前執行個體的 Type

(繼承來源 Object)
MemberwiseClone()

建立目前 Object 的淺層複製。

(繼承來源 Object)
ReleaseLock()

無論執行緒取得鎖定的次數為多少,都會釋放鎖定。

ReleaseReaderLock()

減量鎖定計數。

ReleaseWriterLock()

減量寫入器鎖定上的鎖定計數。

RestoreLock(LockCookie)

將執行緒的鎖定狀態還原到呼叫 ReleaseLock() 之前的狀態。

ToString()

傳回代表目前物件的字串。

(繼承來源 Object)
UpgradeToWriterLock(Int32)

使用逾時值 Int32,將讀取器鎖定升級至寫入器鎖定。

UpgradeToWriterLock(TimeSpan)

使用逾時值 TimeSpan,將讀取器鎖定升級至寫入器鎖定。

適用於

產品 版本
.NET Core 2.0, Core 2.1, Core 2.2, Core 3.0, Core 3.1, 5, 6, 7, 8, 9
.NET Framework 1.1, 2.0, 3.0, 3.5, 4.0, 4.5, 4.5.1, 4.5.2, 4.6, 4.6.1, 4.6.2, 4.7, 4.7.1, 4.7.2, 4.8, 4.8.1
.NET Standard 2.0, 2.1

執行緒安全性

此型別具備執行緒安全。

另請參閱