自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

C#線程同步與死鎖

開發(fā) 后端
本文繼續(xù)C#線程系列講座之四,C#線程同步與死鎖。

在上一講介紹了使用lock來實現(xiàn)C#線程同步。實際上,這個lock是C#的一個障眼法,在C#編譯器編譯lock語句時,將其編譯成了調(diào)用Monitor類。先看看下面的C#源代碼:

  1. public static void MyLock()  
  2. {  
  3.     lock (typeof(Program))  
  4.     {  
  5.     }  
  6. }  
  7.  

上面的代碼通過lock語句使MyLock同步,這個方法被編譯成IL后,代碼如圖1所示。

代碼如圖1 

圖1

從上圖被標注的區(qū)域可以看到,一條lock語句被編譯成了調(diào)用Monitor的Enter和Exit方法。Monitor在System.Threading命名空間中。lock的功能就相當于直接調(diào)用Monitor的Entry方法,所不同的是,lock方法在結束后,會自動解除鎖定,當然,在IL中是調(diào)用了Monitor的Exit方法,但在C#程序中,看起來是自動解鎖的,這類似于C#中的using語句,可以自動釋放數(shù)據(jù)庫等的資源。但如果直接在C#源程序中使用Monitor類,就必須調(diào)用Exit方法來顯式地解除鎖定。如下面的代碼所示:

  1. Monitor.Entry(lockObj);  
  2. try 
  3. {  
  4.     // lockObj的同布區(qū)  
  5. }  
  6. catch(Exception e)  
  7. {  
  8.     // 異常處理代碼  
  9. }  
  10. finally 
  11. {  
  12.     Monitor.Exit(lockObj);  // 解除鎖定  
  13. }  
  14.  

Exit方法最后在finally里調(diào)用,這樣無論在方法在發(fā)生異常、返回還是正常執(zhí)行,都會執(zhí)行到finally,并調(diào)用Exit方法解除鎖定。

Monitor類不僅可以完全取代lock語句(如果只使用lock語句本身的功能,最好還是直接用lock語句吧),還可以使用TryEntry方法設置一個鎖定超時,單位是毫秒。如下面的代碼所示:

  1. if(Monitor.TryEntry(lockObj, 1000))  
  2. {  
  3.     try 
  4.     {  
  5.     }  
  6.     finally 
  7.     {  
  8.         Monitor.Exit(lockObj);  
  9.     }  
  10. }  
  11. else 
  12. {  
  13.     // 超時后的處理代碼  
  14. }  
  15.  

上面的代碼設置了鎖定超時時間為1秒,也就是說,在1秒中后,lockObj還未被解鎖,TryEntry方法就會返回false,如果在1秒之內(nèi),lockObj被解鎖,TryEntry返回true。我們可以使用這種方法來避免死鎖,如下面的代碼所示:

  1. class Program  
  2. {  
  3.     private static Object objA = new Object();  
  4.     private static Object objB = new Object();  
  5.     public static void LockA()  
  6.     {  
  7.         if (Monitor.TryEnter(objA, 1000))  
  8.         {  
  9.             Thread.Sleep(1000);  
  10.             if (Monitor.TryEnter(objB, 2000))  
  11.             {  
  12.                 Monitor.Exit(objB);  
  13.             }  
  14.             else 
  15.             {  
  16.  
  17.                 Console.WriteLine("LockB timeout");  
  18.             }  
  19.             Monitor.Exit(objA);  
  20.         }  
  21.         Console.WriteLine("LockA");  
  22.     }  
  23.     public static void LockB()  
  24.     {  
  25.         if (Monitor.TryEnter(objB, 2000))  
  26.         {  
  27.             Thread.Sleep(2000);  
  28.             if (Monitor.TryEnter(objA, 1000))  
  29.             {  
  30.                 Monitor.Exit(objA);  
  31.             }  
  32.             else 
  33.             {  
  34.                 Console.WriteLine("LockA timeout");  
  35.             }  
  36.             Monitor.Exit(objB);  
  37.         }  
  38.         Console.WriteLine("LockB");  
  39.     }  
  40.     public static void Main()  
  41.     {  
  42.         Thread threadA = new Thread(LockA);  
  43.         Thread threadB = new Thread(LockB);  
  44.         threadA.Start();  
  45.         threadB.Start();  
  46.         Thread.Sleep(4000);           
  47.         Console.WriteLine("線程結束");  
  48.     }  
  49. }  

上面的代碼是在上一講舉的死鎖的例子,但在這一講將lock語句改成了TryEntry方法,而且設置了鎖定超時間,由于在等待一定時間后,不管被鎖定的對象是否被解鎖,TryEntry方法都會返回,因此,上面的代碼是不會死鎖的。運行上面的代碼的結果如圖2所示。

代碼的結果如圖2 

圖2

如果TryEntry方法的超時時間為System.Threading.Timeout.Infinite,TryEntry方法就相當于Entry方法,如果超時時間為0,不管是否解鎖,TryEntry方法都會立即返回。

這樣就解決了C#線程同步與死鎖的問題。

【編輯推薦】

  1. C#自定義特性介紹
  2. C#內(nèi)置特性介紹
  3. 如何進行C#異常類的自定義
  4. C#編程技巧七條
  5. 預測C#與.NET的發(fā)展趨勢
責任編輯:book05 來源: csdn
相關推薦

2009-09-04 14:41:09

C#同步線程

2024-02-27 10:44:58

C#線程后端

2009-08-04 18:00:51

C#線程同步Monitor

2009-09-07 13:19:44

C#線程同步

2011-08-30 15:44:57

C#

2009-07-17 10:37:05

C#多線程

2010-12-21 14:21:36

線程C#

2009-08-18 09:26:07

C#線程功能

2024-10-14 16:25:59

C#線程鎖代碼

2009-08-25 15:09:30

C#線程的啟動

2024-05-17 12:56:09

C#編程線程

2024-05-06 00:00:00

ThreadPool線程調(diào)度

2009-08-12 13:11:24

C#實現(xiàn)遠程線程插入

2009-08-12 18:04:44

編寫C#多線程

2009-08-18 09:33:49

C#特有線程功能

2009-08-28 16:51:32

C#線程控制

2009-09-07 13:02:52

Java和C#線程

2025-02-27 08:15:28

2024-04-23 09:35:27

線程終止C#多線程編程

2012-07-27 10:02:39

C#
點贊
收藏

51CTO技術棧公眾號