C#高级:利用LINQ进行实体列表的集合运算

问题引入:

bash 复制代码
Teacher实体的唯一标识符是Name和Classes字段(或者说这两个字段唯一确定一条数据),如何对两个实体列表做交集、差集运算呢?(并集直接调用AddRange方法即可)

一、重写方法实现

1.原理

重写 Equals() 和 GetHashCode() 方法,使用Intersect() 和 Except() 方法在比较时就会基于这些字段来判断对象是否相等

2.快速生成代码

3.代码

cs 复制代码
public class Teacher
{
    public int Id { get; set; }
    public string Name { get; set; }
    public string Classes { get; set; }

    public override bool Equals(object obj)
    {
        return obj is Teacher teacher &&
               Name == teacher.Name &&
               Classes == teacher.Classes;
    }

    public override int GetHashCode()
    {
        return HashCode.Combine(Name, Classes);
    }
}

class Program
{
    static void Main()
    {
        // 创建两个示例列表
        List<Teacher> listA = new List<Teacher>
        {
            new Teacher { Id = 1, Name = "Alice", Classes = "Math" },
            new Teacher { Id = 2, Name = "Bob", Classes = "Science" },
            new Teacher { Id = 3, Name = "Charlie", Classes = "English" }
        };

        List<Teacher> listB = new List<Teacher>
        {
            new Teacher { Id = 2, Name = "Bob", Classes = "Science" },
            new Teacher { Id = 3, Name = "Charlie", Classes = "English" },
            new Teacher { Id = 4, Name = "David", Classes = "History" }
        };

        // 求交集
        var intersection = listA.Intersect(listB).ToList();

        // 求差集 A - B
        var difference = listA.Except(listB).ToList();


        // 输出结果
        Console.WriteLine("交集:");
        foreach (var teacher in intersection)
        {
            Console.WriteLine($"Id: {teacher.Id}, Name: {teacher.Name}, Classes: {teacher.Classes}");
        }

        Console.WriteLine("\n差集 A - B:");
        foreach (var teacher in difference)
        {
            Console.WriteLine($"Id: {teacher.Id}, Name: {teacher.Name}, Classes: {teacher.Classes}");
        }
    }
}

二、嵌套查询实现

1.原理

cs 复制代码
var intersection = listA.Where(a => listB.Any(b => b.Name == a.Name && b.Classes == a.Classes)).ToList();
var difference = listA.Where(a => !listB.Any(b => b.Name == a.Name && b.Classes == a.Classes)).ToList();

where-Any相当于A表内连接B表( A ∩ B );

where-!Any相当于A表左外连接B表并剔除匹配到的数据(A - B)。

2.代码

cs 复制代码
public class Teacher
{
    public int Id { get; set; }
    public string Name { get; set; }
    public string Classes { get; set; }
}

class Program
{
    static void Main()
    {
        // 创建两个示例列表
        List<Teacher> listA = new List<Teacher>
        {
            new Teacher { Id = 1, Name = "Alice", Classes = "Math" },
            new Teacher { Id = 2, Name = "Bob", Classes = "Science" },
            new Teacher { Id = 3, Name = "Charlie", Classes = "English" }
        };

        List<Teacher> listB = new List<Teacher>
        {
            new Teacher { Id = 2, Name = "Bob", Classes = "Science" },
            new Teacher { Id = 3, Name = "Charlie", Classes = "English" },
            new Teacher { Id = 4, Name = "David", Classes = "History" }
        };

        // 自定义方法来比较 Name 和 Classes
        var intersection = listA.Where(a => listB.Any(b => b.Name == a.Name && b.Classes == a.Classes)).ToList();
        var difference = listA.Where(a => !listB.Any(b => b.Name == a.Name && b.Classes == a.Classes)).ToList();

        // 输出交集
        Console.WriteLine("交集:");
        foreach (var teacher in intersection)
        {
            Console.WriteLine($"Id: {teacher.Id}, Name: {teacher.Name}, Classes: {teacher.Classes}");
        }

        // 输出差集
        Console.WriteLine("\n差集 A - B:");
        foreach (var teacher in difference)
        {
            Console.WriteLine($"Id: {teacher.Id}, Name: {teacher.Name}, Classes: {teacher.Classes}");
        }
    }
}
相关推荐
bugcome_com12 小时前
零基础入门C#:一篇搞懂核心知识点
c#
TTBIGDATA14 小时前
【Atlas】Ambari 中 开启 Kerberos + Ranger 后 Atlas Hook 无权限访问 Kafka Topic:ATLAS_HOOK
大数据·kafka·ambari·linq·ranger·knox·bigtop
程序员敲代码吗15 小时前
如何通过命令行启动COMSOL的参数化、批处理和集群扫描
java·c#·bash
缺点内向17 小时前
C#: 告别繁琐!轻松移除Word文档中的文本与图片水印
c#·自动化·word·.net
喵叔哟18 小时前
06-ASPNETCore-WebAPI开发
服务器·后端·c#
2501_9307077818 小时前
使用 C# .NET 从 PowerPoint 演示文稿中提取背景图片
c#·powerpoint·.net
初级代码游戏18 小时前
套路化编程 C# winform 自适应缩放布局
开发语言·c#·winform·自动布局·自动缩放
大空大地202620 小时前
流程控制语句--switch多分支语句使用、while循环语句的使用、do...while语句、for循环
c#
kylezhao201921 小时前
C#序列化与反序列化详细讲解与应用
c#
JQLvopkk21 小时前
C# 实践AI :Visual Studio + VSCode 组合方案
人工智能·c#·visual studio