培训
集合和数据结构
类似的数据在作为集合而存储和操作时通常可以得到更高效地处理。 可以使用 System.Array 类或 System.Collections、System.Collections.Generic、System.Collections.Concurrent 和 System.Collections.Immutable 命名空间中的类来添加、删除和修改集合中的任一单个元素或某个范围的元素。
有两种主要的集合类型:泛型集合和非泛型集合。 泛型集合在编译时是类型安全的。 因此,泛型集合通常能提供更好的性能。 构造泛型集合时,它们接受类型形参;并在向该集合添加项或从该集合删除项时无需在 Object 类型间来回转换。 此外,Windows 应用商店应用支持大多数泛型集合。 非泛型集合将项存储为 Object,需要强制转换,并且大多数不支持 Windows 应用商店应用开发。 但是,你可能会看到在较旧的代码中有非泛型集合。
自 .NET Framework 4 起,System.Collections.Concurrent 命名空间中的集合可提供高效的线程安全操作,以便从多个线程访问集合项。 System.Collections.Immutable 命名空间(NuGet 包)中的不可变集合类本质上就是线程安全的,因为操作在原始集合的副本上进行且不能修改原始集合。
所有集合都提供用于在集合中添加、删除或查找项的方法。 此外,所有直接或间接实现 ICollection 接口或 ICollection<T> 接口的集合均共享这些功能:
可枚举集合
.NET 集合实现 System.Collections.IEnumerable 或 System.Collections.Generic.IEnumerable<T>,以启用要循环访问的集合。 可将枚举器看作集合中可指向任何元素的可移动指针。 foreach, in 语句和 For Each...Next 语句使用 GetEnumerator 方法公开的枚举器并隐藏操作枚举器的复杂性。 此外,任何实现 System.Collections.Generic.IEnumerable<T> 的集合均被认为是可查询类型,并可使用 LINQ 对其进行查询。 LINQ 查询提供数据访问的一个通用模式。 它们通常比标准
foreach
循环更简洁、更具可读性,并提供筛选、排序和分组功能。 LINQ 查询还可提高性能。 有关详细信息,请参阅 LINQ to Objects (C#)、LINQ to Objects (Visual Basic)、并行 LINQ (PLINQ)、LINQ 查询 (C#) 简介和基本查询操作 (Visual Basic)。可将集合内容复制到数组
可使用 CopyTo 方法将所有集合复制到数组中;但新数组中的元素顺序是以枚举器返回元素的顺序为依据。 得到的数组始终是一维的,下限为零。
此外,许多集合类包含下列功能:
容量和计数属性
集合的容量是它可包含的元素数。 集合的计数是它实际所含的元素数。 某些集合隐藏容量、计数或将这两者都隐藏。
达到当前容量时,大多数集合会自动扩展容量。 重新分配内存并将元素从旧集合复制到新集合。 这减少了要求使用集合的代码;但集合的性能可能会受到不利影响。 例如,对 List<T> 来说,如果 Count 比 Capacity 少,那么添加项就是一项 O(1) 操作。 如需增加容量以容纳新元素,则添加项成为 O(
n
) 操作,其中n
是 Count。 避免因多次重新分配而导致的性能较差的最佳方式是:将初始容量设置为集合的估计大小。BitArray 是一种特殊情况;它的容量与其长度相同,而其长度与其计数相同。
下限一致
集合的下限是其第一个元素的索引。 System.Collections 命名空间中的所有索引集合的下限均为零,这表示它们从 0 开始建立索引。 Array 默认下限为零,但使用 Array.CreateInstance 创建 Array 类的实例时可定义其他下限。
同步以从多个线程进行访问(仅 System.Collections 类)。
System.Collections 命名空间中的非泛型集合类型通过同步提供一些线程安全性;通常通过 SyncRoot 和 IsSynchronized 成员公开。 这些集合不是默认为线程安全的。 如需对集合进行可扩展、高效的多线程访问,请使用 System.Collections.Concurrent 命名空间中的一个类或考虑使用不可变集合。 有关详细信息,请参阅线程安全集合。
一般情况下,应使用泛型集合。 下表介绍了一些常用的集合方案和可用于这些方案的集合类。 如果你是使用泛型集合的新手,此表将帮助你选择最适合你的任务的泛型集合。
我要…… | 泛型集合选项 | 非泛型集合选项 | 线程安全或不可变集合选项 |
---|---|---|---|
将项存储为键/值对以通过键进行快速查找 | Dictionary<TKey,TValue> | Hashtable (根据键的哈希代码组织的键/值对的集合。) |
ConcurrentDictionary<TKey,TValue> ReadOnlyDictionary<TKey,TValue> ImmutableDictionary<TKey,TValue> |
按索引访问项 | List<T> | Array ArrayList |
ImmutableList<T> ImmutableArray |
使用项先进先出 (FIFO) | Queue<T> | Queue | ConcurrentQueue<T> ImmutableQueue<T> |
使用数据后进先出 (LIFO) | Stack<T> | Stack | ConcurrentStack<T> ImmutableStack<T> |
按顺序访问项 | LinkedList<T> | 无建议 | 无建议 |
删除集合中的项或向集合添加项时接收通知。 (实现 INotifyPropertyChanged 和 INotifyCollectionChanged) | ObservableCollection<T> | 无建议 | 无建议 |
已排序的集合 | SortedList<TKey,TValue> | SortedList | ImmutableSortedDictionary<TKey,TValue> ImmutableSortedSet<T> |
数学函数的一个集 | HashSet<T> SortedSet<T> |
无建议 | ImmutableHashSet<T> ImmutableSortedSet<T> |
选择集合类时,应该考虑可能牺牲的性能。 使用下表来参考各种可变集合类型在算法复杂性方面与其对应不可变对应项的比较情况。 不可变的集合类型通常性能较低,但却提供了不可变性,这通常是一种非常有效的优点。
可变 | 分期 | 最坏情况 | 不可变 | 复杂性 |
---|---|---|---|---|
Stack<T>.Push |
O(1) | O(n ) |
ImmutableStack<T>.Push |
O(1) |
Queue<T>.Enqueue |
O(1) | O(n ) |
ImmutableQueue<T>.Enqueue |
O(1) |
List<T>.Add |
O(1) | O(n ) |
ImmutableList<T>.Add |
O(log n ) |
List<T>.Item[Int32] |
O(1) | O(1) | ImmutableList<T>.Item[Int32] |
O(log n ) |
List<T>.Enumerator |
O(n ) |
O(n ) |
ImmutableList<T>.Enumerator |
O(n ) |
HashSet<T>.Add , lookup |
O(1) | O(n ) |
ImmutableHashSet<T>.Add |
O(log n ) |
SortedSet<T>.Add |
O(log n ) |
O(n ) |
ImmutableSortedSet<T>.Add |
O(log n ) |
Dictionary<T>.Add |
O(1) | O(n ) |
ImmutableDictionary<T>.Add |
O(log n ) |
Dictionary<T> lookup |
O(1) | O(1) - 或者从严格意义上说,O(n ) |
ImmutableDictionary<T> lookup |
O(log n ) |
SortedDictionary<T>.Add |
O(log n ) |
O(n log n ) |
ImmutableSortedDictionary<T>.Add |
O(log n ) |
可以使用 for
循环或 foreach
循环来有效地枚举 List<T>
。 但是,由于其索引器的 O(log n
) 时间,ImmutableList<T>
在 for
循环内的效果较差。 使用 foreach
循环枚举 ImmutableList<T>
很有效,因为 ImmutableList<T>
使用二进制树来存储其数据,而不是像 List<T>
那样使用简单数组。 数组可以非常快速地编入索引,但必须向下遍历二进制树,直到找到具有所需索引的节点。
此外,SortedSet<T>
与 ImmutableSortedSet<T>
的复杂性相同。 这是因为它们都使用了二进制树。 当然,显著的差异在于 ImmutableSortedSet<T>
使用不可变的二进制树。 由于 ImmutableSortedSet<T>
还提供了一个允许变化的 System.Collections.Immutable.ImmutableSortedSet<T>.Builder 类,因此可以同时实现不可变性和保障性能。
Title | 描述 |
---|---|
选择集合类 | 描述不同的集合并帮助你为你的方案选择一个集合。 |
常用的集合类型 | 描述诸如 System.Array、System.Collections.Generic.List<T> 和 System.Collections.Generic.Dictionary<TKey,TValue> 等常用泛型和非泛型集合类型。 |
何时使用泛型集合 | 讨论泛型集合类型的使用。 |
集合内的比较和排序 | 讨论在集合中使用等同性比较和排序比较。 |
已排序的集合类型 | 描述已排序集合的性能和特征 |
哈希表和字典集合类型 | 描述泛型和非泛型基于哈希的字典类型的功能。 |
线程安全集合 | 介绍支持从多个线程进行安全有效的并发访问的集合类型,例如 System.Collections.Concurrent.BlockingCollection<T> 和 System.Collections.Concurrent.ConcurrentBag<T>。 |
System.Collections.Immutable | 介绍不可变集合并提供各集合类型的链接。 |
System.Array System.Collections System.Collections.Concurrent System.Collections.Generic System.Collections.Specialized System.Linq System.Collections.Immutable