C# 从 List 中移除另一个集合

在 C# 中,从一个集合中移除另一个集合的元素,以及在不影响原集合的情况下返回新集合,有多种实现方式。我来详细解释并举例说明。

一、从 List 中移除另一个集合

1. 使用 RemoveAll + Contains(修改原集合)

csharp 复制代码
List<int> list1 = new List<int> { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };
List<int> list2 = new List<int> { 2, 4, 6, 8, 10 };

// 移除 list1 中存在于 list2 的所有元素(修改原集合)
list1.RemoveAll(item => list2.Contains(item));

Console.WriteLine(string.Join(", ", list1)); // 输出:1, 3, 5, 7, 9

优点 :简单直接
缺点 :对每个元素都调用 Contains,大数据量时性能较差(O(n*m))

2. 使用 Except(返回新集合,不影响原集合)

csharp 复制代码
List<int> list1 = new List<int> { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };
List<int> list2 = new List<int> { 2, 4, 6, 8, 10 };

// 返回新集合,原集合不变
List<int> result = list1.Except(list2).ToList();

Console.WriteLine("原集合:" + string.Join(", ", list1));  // 输出:1, 2, 3, 4, 5, 6, 7, 8, 9, 10
Console.WriteLine("新集合:" + string.Join(", ", result)); // 输出:1, 3, 5, 7, 9

优点 :不影响原集合,使用 HashSet 优化,性能好(O(n+m))
缺点:会去重(如果原集合有重复元素,结果中只保留一个)

3. 使用 HashSet 优化性能(修改原集合)

csharp 复制代码
List<int> list1 = new List<int> { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };
List<int> list2 = new List<int> { 2, 4, 6, 8, 10 };

HashSet<int> set = new HashSet<int>(list2);
list1.RemoveAll(item => set.Contains(item));

Console.WriteLine(string.Join(", ", list1)); // 输出:1, 3, 5, 7, 9

优点 :HashSet 的 Contains 是 O(1),大数据量性能好(O(n+m))
缺点:需要额外内存

二、返回新集合不影响原集合的多种方式

1. 使用 Except 方法(最简单)

csharp 复制代码
List<string> fruits = new List<string> { "苹果", "香蕉", "橙子", "葡萄", "西瓜", "香蕉" };
List<string> toRemove = new List<string> { "香蕉", "葡萄" };

// 返回新集合,原集合不变(但会去重)
List<string> result = fruits.Except(toRemove).ToList();

Console.WriteLine("原集合:" + string.Join(", ", fruits));  // 苹果, 香蕉, 橙子, 葡萄, 西瓜, 香蕉
Console.WriteLine("新集合:" + string.Join(", ", result)); // 苹果, 橙子, 西瓜

2. 使用 Where + Contains(保留重复元素)

csharp 复制代码
List<string> fruits = new List<string> { "苹果", "香蕉", "橙子", "葡萄", "西瓜", "香蕉" };
List<string> toRemove = new List<string> { "香蕉", "葡萄" };

// 返回新集合,原集合不变,保留重复元素
List<string> result = fruits.Where(f => !toRemove.Contains(f)).ToList();

Console.WriteLine("原集合:" + string.Join(", ", fruits));  // 苹果, 香蕉, 橙子, 葡萄, 西瓜, 香蕉
Console.WriteLine("新集合:" + string.Join(", ", result)); // 苹果, 橙子, 西瓜

3. 使用 HashSet 优化的版本(推荐)

csharp 复制代码
List<string> fruits = new List<string> { "苹果", "香蕉", "橙子", "葡萄", "西瓜", "香蕉" };
List<string> toRemove = new List<string> { "香蕉", "葡萄" };

// 使用 HashSet 提升性能
HashSet<string> removeSet = new HashSet<string>(toRemove);
List<string> result = fruits.Where(f => !removeSet.Contains(f)).ToList();

Console.WriteLine("原集合:" + string.Join(", ", fruits));  // 苹果, 香蕉, 橙子, 葡萄, 西瓜, 香蕉
Console.WriteLine("新集合:" + string.Join(", ", result)); // 苹果, 橙子, 西瓜

三、自定义对象的示例

csharp 复制代码
public class Person
{
    public int Id { get; set; }
    public string Name { get; set; }
}

List<Person> employees = new List<Person>
{
    new Person { Id = 1, Name = "张三" },
    new Person { Id = 2, Name = "李四" },
    new Person { Id = 3, Name = "王五" },
    new Person { Id = 4, Name = "赵六" }
};

List<Person> toRemove = new List<Person>
{
    new Person { Id = 2, Name = "李四" },
    new Person { Id = 4, Name = "赵六" }
};

// 方法1:按Id移除
var removeIds = new HashSet<int>(toRemove.Select(p => p.Id));
List<Person> result = employees.Where(e => !removeIds.Contains(e.Id)).ToList();

// 方法2:使用 Except(需要实现 IEquatable<Person> 或传入比较器)
// 先实现比较器
public class PersonComparer : IEqualityComparer<Person>
{
    public bool Equals(Person x, Person y) => x.Id == y.Id && x.Name == y.Name;
    public int GetHashCode(Person obj) => obj.Id.GetHashCode() ^ obj.Name.GetHashCode();
}

// 使用比较器
List<Person> result2 = employees.Except(toRemove, new PersonComparer()).ToList();

四、性能对比和选择建议

方法 是否修改原集合 性能 是否去重 适用场景
RemoveAll + Contains O(n*m) 差 保留重复 数据量小
RemoveAll + HashSet O(n+m) 好 保留重复 修改原集合
Except O(n+m) 好 会去重 需要去重
Where + Contains O(n*m) 差 保留重复 数据量小
Where + HashSet O(n+m) 好 保留重复 推荐

推荐做法

  • 需要修改原集合:RemoveAll + HashSet
  • 不需要修改原集合:Where + HashSet
  • 简单场景且数据量小:直接用 Except
相关推荐
大空大地20261 小时前
C#进阶语法**总结
c#
ez52fF0k81 小时前
.NET11云原生CI/CD在云原生应用持续集成与交付安全加固
前端·c#·交互
qxl_7999153 小时前
Windows 显卡掉线无报警|模型推理全套防呆方案(实操完整版)
windows·stm32·单片机·推理显卡掉线误报警防呆
没什么本事13 小时前
关于C# panel 添加lable问题 -- 明确X和Y 位置错误
android·java·c#
火星papa15 小时前
C# 实现平滑流畅的进度条ProgressBar
c#·进度条·progressbar·平滑流畅
游乐码16 小时前
UnityGUI(五)GUI控件综合使用
开发语言·unity·c#
程序leo源16 小时前
C语言知识总结
c语言·开发语言·c++·经验分享·笔记·青少年编程·c#
数智工坊17 小时前
VMware 17 Pro 中 Ubuntu 虚拟机共享 Windows 文件夹(完美踩坑版)
linux·人工智能·windows·ubuntu
烛阴17 小时前
TEngine 入门系列(二):三件套环境搭建 -- Unity + TEngine + AI 助手
前端·c#·unity3d