news 2026/6/21 4:11:25

C# 集合全面指南:种类、遍历、用法与注意事项

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
C# 集合全面指南:种类、遍历、用法与注意事项

一、C# 集合概述

C# 中的集合是用于存储和管理一组相关对象的特殊类,它们提供了比数组更强大的功能,如动态大小调整、排序、搜索等。

二、集合的主要种类

1.非泛型集合 (System.Collections)

// 已过时,不推荐在新项目中使用 ArrayList list = new ArrayList(); list.Add(1); // 可以添加任何类型 list.Add("string"); // 类型不安全

2.泛型集合 (System.Collections.Generic)-推荐使用

列表类集合
// List<T> - 最常用的动态数组 List<string> fruits = new List<string> { "Apple", "Banana", "Orange" }; fruits.Add("Mango"); fruits.Remove("Banana"); // LinkedList<T> - 双向链表 LinkedList<int> numbers = new LinkedList<int>(); numbers.AddLast(1); numbers.AddFirst(0); // ObservableCollection<T> - 支持数据绑定和通知 ObservableCollection<string> items = new ObservableCollection<string>(); items.CollectionChanged += (sender, e) => { Console.WriteLine("集合已更改"); };
字典类集合
// Dictionary<TKey, TValue> - 哈希表实现的键值对 Dictionary<string, int> ages = new Dictionary<string, int> { ["Alice"] = 25, ["Bob"] = 30 }; ages["Charlie"] = 28; // SortedDictionary<TKey, TValue> - 基于二叉搜索树的有序字典 SortedDictionary<int, string> sortedDict = new SortedDictionary<int, string>(); // ConcurrentDictionary<TKey, TValue> - 线程安全的字典 ConcurrentDictionary<string, int> concurrentDict = new ConcurrentDictionary<string, int>();
队列和栈
// Queue<T> - 先进先出(FIFO) Queue<string> queue = new Queue<string>(); queue.Enqueue("First"); queue.Enqueue("Second"); string firstItem = queue.Dequeue(); // Stack<T> - 后进先出(LIFO) Stack<int> stack = new Stack<int>(); stack.Push(1); stack.Push(2); int topItem = stack.Pop();
集合类
// HashSet<T> - 不包含重复元素的高性能集合 HashSet<int> uniqueNumbers = new HashSet<int> { 1, 2, 2, 3 }; // 结果: 1, 2, 3 // SortedSet<T> - 有序的不重复集合 SortedSet<string> sortedSet = new SortedSet<string>(); // ReadOnlyCollection<T> - 只读集合包装器 IReadOnlyList<string> readOnlyList = fruits.AsReadOnly();

不可变集合 (System.Collections.Immutable)

// 线程安全且不可变的集合 ImmutableList<string> immutableList = ImmutableList.Create("A", "B", "C"); ImmutableDictionary<string, int> immutableDict = ImmutableDictionary.Create<string, int>();

三、集合的遍历方式

1.foreach 循环 (推荐)

foreach (var fruit in fruits) { Console.WriteLine(fruit); } // 遍历字典 foreach (KeyValuePair<string, int> kvp in ages) { Console.WriteLine($"{kvp.Key}: {kvp.Value}"); }

2.for 循环 (适用于需要索引的情况)

for (int i = 0; i < fruits.Count; i++) { Console.WriteLine($"索引 {i}: {fruits[i]}"); }

四、常见用法示例

1.集合初始化

// 多种初始化方式 List<int> numbers1 = new List<int> { 1, 2, 3 }; List<int> numbers2 = new() { 1, 2, 3 }; // C# 9.0 var numbers3 = new List<int> { 1, 2, 3 }; // 字典初始化 var dict1 = new Dictionary<string, int> { { "Alice", 25 }, { "Bob", 30 } }; var dict2 = new Dictionary<string, int> { ["Alice"] = 25, ["Bob"] = 30 };

2.集合操作

// 添加元素 fruits.Add("Grape"); fruits.AddRange(new[] { "Peach", "Pear" }); fruits.Insert(1, "Lemon"); // 删除元素 fruits.Remove("Apple"); fruits.RemoveAt(0); fruits.RemoveAll(f => f.StartsWith("A")); // 查找元素 bool hasApple = fruits.Contains("Apple"); int index = fruits.IndexOf("Banana"); var apple = fruits.Find(f => f == "Apple"); // 排序 fruits.Sort(); fruits.Sort((x, y) => x.Length.CompareTo(y.Length)); // 转换数组 string[] fruitArray = fruits.ToArray(); List<string> fruitList = fruitArray.ToList();

3.性能优化操作

// 预设容量提升性能 List<string> largeList = new List<string>(10000); // 批量操作 List<int> source = Enumerable.Range(1, 10000).ToList(); List<int> destination = new List<int>(source.Count); destination.AddRange(source);

五、重要注意事项

1.集合选择指南

选择原则: - 需要快速随机访问 → List<T> - 频繁插入/删除 → LinkedList<T> - 键值对存储 → Dictionary<TKey, TValue> - 需要排序 → SortedDictionary<TKey, TValue> - 去重 → HashSet<T> - 线程安全 → ConcurrentBag/Queue/Stack/Dictionary - 只读 → ReadOnlyCollection<T> - 不可变 → ImmutableList/Dictionary

2.遍历时修改集合的陷阱

// 遍历时直接修改会抛出InvalidOperationException foreach (var fruit in fruits) { if (fruit == "Apple") fruits.Remove(fruit); // 运行时错误! } // 正确做法1:使用ToArray()副本 foreach (var fruit in fruits.ToArray()) { if (fruit == "Apple") fruits.Remove(fruit); } // 正确做法2:使用for循环反向遍历 for (int i = fruits.Count - 1; i >= 0; i--) { if (fruits[i] == "Apple") fruits.RemoveAt(i); } // 正确做法3:先标记后删除 var itemsToRemove = new List<string>(); foreach (var fruit in fruits) { if (fruit == "Apple") itemsToRemove.Add(fruit); } foreach (var item in itemsToRemove) { fruits.Remove(item); }

3.字典使用注意事项

Dictionary<string, int> dict = new Dictionary<string, int>(); // 错误:访问不存在的键 int value = dict["nonexistent"]; // KeyNotFoundException // 正确:使用TryGetValue if (dict.TryGetValue("Alice", out int age)) { Console.WriteLine(age); } // 正确:使用ContainsKey检查 if (dict.ContainsKey("Alice")) { value = dict["Alice"]; } // 字典键的自定义类型需要正确实现GetHashCode和Equals public class Person { public string Name { get; set; } public override bool Equals(object obj) => obj is Person other && Name == other.Name; public override int GetHashCode() => Name?.GetHashCode() ?? 0; }

4.性能注意事项

// List<T>的容量管理 List<int> list = new List<int>(); for (int i = 0; i < 1000; i++) { list.Add(i); // 多次重新分配内存 // 如果知道大小,预设容量 List<int> optimizedList = new List<int>(1000); } // HashSet<T> vs List<T>的Contains性能 HashSet<string> hashSet = new HashSet<string>(); // O(1) List<string> normalList = new List<string>(); // O(n)

5.线程安全

// 非线程安全 List<string> unsafeList = new List<string>(); // 使用锁 object lockObj = new object(); lock (lockObj) { unsafeList.Add("item"); } // 使用并发集合 ConcurrentBag<string> safeBag = new ConcurrentBag<string>(); safeBag.Add("item");

6.内存管理

// 及时清理不再使用的集合 List<byte[]> largeData = new List<byte[]>(); // 使用后清理 largeData.Clear(); largeData.TrimExcess(); // 释放多余容量 // 使用using语句处理需要释放资源的集合 using (BlockingCollection<int> collection = new BlockingCollection<int>()) { // 使用集合 }

7.空值处理

// 处理可能的null集合 List<string> possibleNullList = GetList(); var count = possibleNullList?.Count ?? 0; // 安全访问 // 使用空集合代替null public List<string> GetItems() { return new List<string>(); // 而不是返回null }

六、总结

  1. 优先使用泛型集合,避免非泛型集合

  2. 根据场景选择合适的集合类型

  3. 遍历时避免修改集合内容

  4. 合理预设集合容量提升性能

  5. 及时释放不再使用的集合

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/16 17:42:15

Excalidraw与Figma对比:UI设计初期哪个更强?

Excalidraw 与 Figma&#xff1a;谁才是设计初期的真正利器&#xff1f; 在一次产品需求评审会上&#xff0c;团队正为新功能的交互流程争论不休。产品经理在白板上画了几笔草图&#xff0c;工程师指着某个箭头说“这里逻辑不通”&#xff0c;设计师立刻拖动方框重新排布——十…

作者头像 李华
网站建设 2026/6/19 20:05:20

Excalidraw图形容器:分组管理复杂结构

Excalidraw图形容器&#xff1a;分组管理复杂结构 在一场远程架构评审会议上&#xff0c;团队正围绕一个微服务系统的草图激烈讨论。屏幕上布满了散落的矩形、箭头和文字标签——有人刚修改了“订单服务”的位置&#xff0c;结果不小心拖动了本属于“支付网关”的连接线&#x…

作者头像 李华
网站建设 2026/6/19 13:51:42

AI原生应用领域多租户的性能监控指标与方法

AI原生应用领域多租户的性能监控指标与方法关键词&#xff1a;AI原生应用、多租户架构、性能监控、指标体系、云原生技术摘要&#xff1a;随着AI技术与云原生架构的深度融合&#xff0c;"AI原生应用"已成为企业智能化转型的核心载体。这类应用的典型特征是支持多租户…

作者头像 李华
网站建设 2026/6/20 11:46:20

Excalidraw自定义素材库:建立专属图形资源中心

Excalidraw自定义素材库&#xff1a;建立专属图形资源中心 在技术团队日益依赖可视化协作的今天&#xff0c;一张清晰、一致且高效的架构图&#xff0c;往往比千言万语更能推动项目前进。然而现实是&#xff0c;每次画图都像是从零开始——Redis图标画得不像上次&#xff0c;微…

作者头像 李华
网站建设 2026/6/18 20:56:11

Unity3D 语音操控效果演示

基于 Unity3D 引擎实现语音控制的模型动画切换系统。自动识别麦克风并解析语音指令&#xff0c;如跳跃、奔跑、换弹、射击、待机等&#xff0c;使 3D 模型实时切换对应动画。同时支持场景切换与程序退出等功能。 Unity3D 语音操控效果演示

作者头像 李华
网站建设 2026/6/18 14:36:10

Excalidraw离线使用指南:无网络环境下的应对策略

Excalidraw离线使用指南&#xff1a;无网络环境下的应对策略 在金融系统架构评审会上&#xff0c;投影仪突然断网&#xff0c;白板上的微服务拓扑图再无法同步更新&#xff1b;野外勘探队的工程师试图用AI生成井场布局草图&#xff0c;却因卫星信号中断而被迫中止——这些场景暴…

作者头像 李华