C#中實現(xiàn)線程同步lock關鍵字的用法詳解
1. lock關鍵字保證一個代碼塊在執(zhí)行的過程中不會受到其他線程的干擾,這是通過在該代碼塊的運行過程中對特定的對象加互斥鎖來實現(xiàn)的。
2. lock關鍵字的參數(shù)必須是引用類型的對象。lock對基本數(shù)據(jù)類型如int,long等無效,因為它所作用的類型必須是對象。如果傳入long類型數(shù)據(jù),勢必被轉換為Int64結構類型,則加鎖的是全新的對象引用。如果需要對它們進行互斥訪問限制,可以使用System.Threading.Interlocked類提供的方法,這個類是提供原子操作的。
3. lock(this)的使用要慎重。共有類型中使用lock(this),如果新的對象被創(chuàng)建并加鎖,極易造成死鎖。
4. 鎖定ICollection類型對象時,應lock其SyncRoot屬性。
SyncRoot屬性在接口ICollection中聲明,其實現(xiàn)方式各不相同。
例如在Collection(System.Collections.ObjectModel)中實現(xiàn)如下:
object ICollection.SyncRoot { get { if (this._syncRoot == null) { ICollection items = this.items as ICollection; if (items != null) { this._syncRoot = items.SyncRoot; } else { Interlocked.CompareExchange(ref this._syncRoot, new object(), null); } } return this._syncRoot; } }
而在List<T>,ArrayList等類中實現(xiàn)如下:
object ICollection.SyncRoot { get { if (this._syncRoot == null) { Interlocked.CompareExchange(ref this._syncRoot, new object(), null); } return this._syncRoot; } }
在Array類中則直接返回了this:
public object SyncRoot { get { return this; } }
5. lock關鍵字是用Monitor(管程)類實現(xiàn)的
lock(x) { DoSomething(); }
System.Object obj = (System.Object)x; System.Threading.Monitor.Enter(obj); try { DoSomething(); } finally { System.Threading.Monitor.Exit(obj); }
以上兩段代碼是等效的。(MSDN)
使用lock關鍵字相對于Monitor類在使用上更簡單,也更加保險。