C#5.0泛型集合類型簡述

來源:互聯網
上載者:User

標籤:

.NET中的泛型集合

在這裡主要介紹常見的泛型集合,很多時候其並發時的執行緒安全性常常令我們擔憂。因而簡述下.NET並發時安全執行緒特性,其詳情請見MSDN。

  • 普通集合都不支援多重並發寫操作
  • 部分支援單線程寫和並發讀操作
  • 同時.NET4添加了大量並發集合

 

首先介紹常見的泛型集合介面,其大部分都位於System.Collection.Generic命名空間。

  • IEnumerable<T>,其可以擷取一個IEnumerator<T>迭代器,如果從資料庫的角度來看,前者是表,後者是遊標,同時這兩個介面是唯一具有可變性的集合介面。
  • ICollection<T>,它擴充了IEnumerable<T>,添加了Count和IsReadOnly屬性,Add和Remove等操作方法,Contains等判定函數,所有的標準泛型集合都實現了該介面。
  • IList<T>,提供定位功能,包括一個索引器、Insert和RemoveAt,我們通常認為可以通過索引對該泛型集合進行隨機訪問。、
  • IDictionary<TKey, TValue>,表示索引值對集合,擴充了ICollection<KeyValuePair<TKey, TValue>>,取值可以用TryXXX方式。
  • ISet<T>表示唯一值集,包含大量集合操作:交、並、補。

 

接下來介紹具體的集合泛型集合類型,在實際中需要根據具體情境選擇最適合的集合類型。

  • List<T>,其是列表的預設選擇,內含一個數組,並且提供列表的邏輯大小Count和後台數組的大小Capacity,當數組滿了時,會進行擴容。由於是連續型的資料結構,其添加刪除操作的成本較高,提供二分尋找,尋找效率高。同時,其Sort操作會修改原始列表的內容,與OrderBy不同,並且Sort是不穩定的,會出現相等元素順序不同的情況。
  • 數組,最基礎的集合,均派生自System.Array,包括一維數組T[10],二維數組T[10, 20]等,通過Array類的靜態方法進行ConvertAll、FindAll和BinarySearch等操作。
  • Colletion<T>,位於System.Colletion.ObjectModel命名空間,為BindingList<T>和ObservableCollection<T>等擴充類型提供基類。與雙向繫結相關的集合類型,注意它們只會在封裝器發生變化發出通知,而基礎列表改變時不會引發任何事件。
  • ReadOnlyCollection<T>和ReadOnlyObservableCollection<T>,其也類似於封裝器,後者實現了INotifyCollectionChanged, INotifyPropertyChanged兩個介面。
  • Dictionary<TKey, TValue>,使用散列表,尋找效能的優劣取決於散列函數的優劣,預設使用Equals和GetHashCode,可以通過制定IEqualityComparer<TKey>作為參數。
  • SortList<TKey, TValue>和SortedDictionary<TKey, TValue>,兩者都是字典類,前者內部維護一個排序的數組,添加刪除操作的事件複雜度為O(n),後者內部維護一個紅/黑樹狀結構,添加刪除操作事件複雜度為O(log n),但會消耗更多的堆記憶體,使用IComparer<TKey>作比較。
  • HashSet<T>,是不含值的Dictionary<,>,具有相同效能特性,並且所維護順序一般與添加順序無關。
  • SortedSet<T>,是沒有值得SortedDictionary<,>,維護一個紅/黑樹狀結構,添加刪除和檢查操作的事件複雜度為O(log n)。提供GetViewBetween方法返回介於原創組上下限之間的另一個SortedSet<T>,注意這是一個動態視圖,會隨著原創組的改變而改變。儘管看起來很方便,但需要注意的是"天下沒有免費的午餐",為保持內部一致性,操作的代價更大。
  • Queue<T>,構建一個環形緩衝區,實際維護一個基礎數組,包含兩個索引,分別記住入隊和出隊的位置(Slot),如果入隊指標追上出隊指標,則進行擴容。提供Enqueue、Dequeue、Peek等方法進行入隊、出隊、查看操作。
  • Stack<T>,其實現更簡單,可以看做是一個提供Push、Pop、Peek操作的List<T>。

 

最後介紹並行集合,也就是安全執行緒的集合。(注意所有的並發類型都未實現IList<T>介面)

  • IProducerConsumerCollection<T>和BlockingCollection<T>,前者是生產者/消費者模型中資料存放區的抽象,後者是其封裝類,使用ConcurrentQueue<T>作為後台儲存,提供ToArray方法獲得集合目前狀態快照,TryXXX方法允許有效失敗模式減少對鎖的需求。(例如,當隊列中只有一個項時,兩個線程同時判斷它是否有項,並且都返回true,這是一個線程執行了出隊操作,而另外一個線程在執行出隊操作時,將拋出異常,因而需要對驗證隊列是否有項操作和有項就出隊操作作為一個整體,需要添加鎖)
  • ConcurrentBag<T>,ConcurrentQueue<T>,ConcurrentStack<T>,它們是對IProducerConsumerCollection<T>的實現,其GetEnumerator()方法返回集合快照,迭代時可以改變集合,但該改變不會反應到迭代器中。
  • ConcurrentDictionary<TKey, TValue>, 實現了IDictionary<TKey, TValue>介面。支援並發的讀寫和安全執行緒的迭代,但不同是,其在迭代過程中對字典的改變不能確定是否反應到迭代器上。

 

小節:在日常工作中,當遇到需要並行作業非集合類型的全域變數時,需要使用鎖來處理;而當是集合類型時,就需要使用對應的並行集合類來處理,其能很好的TPL協作在一起。尤其在使用非安全執行緒的字典類進行並行作業時,有時會出現死迴圈等情形,尤其需要注意。

 

參考文獻

  1. Jon, Skeet. 深入理解C#(第3版)[M]. 北京:人民郵電出版社, 2014. 469-483

C#5.0泛型集合類型簡述

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.