比较两个集合值c#

时间:2017-07-14 09:08:11

标签: c# .net generics collections reference

我有两个具有相同值的集合,但它们具有不同的引用。在没有foreach语句的情况下比较两个集合的最佳方法是什么, 以下是我创建的示例应用程序

using System;
using System.Collections.Generic;
using System.Collections.ObjectModel;
using System.Linq;

namespace CollectionComparer
{
  public class Program
  {
    private static void Main(string[] args)
    {
        var persons = GetPersons();
        var p1 = new ObservableCollection<Person>(persons);
        IList<Person> p2 = p1.ToList().ConvertAll(x =>
            new Person
            {
                Id = x.Id,
                Age = x.Age,
                Name = x.Name,
                Country = x.Country
            });

        //p1[0].Name = "Name6";
        //p1[1].Age = 36;

        if (Equals(p1, p2))
            Console.WriteLine("Collection and its values are Equal");
        else
            Console.WriteLine("Collection and its values are not Equal");
        Console.ReadLine();
    }

    public static IEnumerable<Person> GetPersons()
    {
        var persons = new List<Person>();
        for (var i = 0; i < 5; i++)
        {
            var p = new Person
            {
                Id = i,
                Age = 20 + i,
                Name = "Name" + i,
                Country = "Country" + i
            };
            persons.Add(p);
        }
        return persons;
    }
  }
}

public class Person
{
  public int Id { get; set; }
  public string Name { get; set; }
  public int Age { get; set; }
  public string Country { get; set; }
}

在上面的代码中,我需要比较集合p1和p2。但结果始终是“收集及其价值不相等”,因为两个收集都有不同的参考。 有没有一种通用的方法可以在不使用foreach和比较类型特定属性的情况下进行这种比较。

2 个答案:

答案 0 :(得分:4)

您可以使用Enumerable.SequenceEqual<T>(IEnumerabe<T> first, IEnumerbale<T> second)

这将按顺序比较两个序列 ,如果其中包含的项目相同且两者具有相同数量的元素,则返回true

一个警告;因为Person没有覆盖Equals(object obj)SequenceEqual会在比较任意两个Person对象时执行默认的引用相等性检查,并且您可能想要值平等语义。要解决这个问题,请至少覆盖bool Equals(object obj)int GetHashCode()(同时考虑实施IEquatable<Person>是一种很好的做法):

public class Person
{
    public int Id { get; set; }
    public string Name { get; set; }
    public int Age { get; set; }
    public string Country { get; set; }

    public override bool Equals(object obj)
    {
         var person = obj as Person;

         if (person == null) return false;

         return person.Id == Id && person.Name = Name && //etc.
    }

    public override int GetHashCode() 
        => Id.GetHashCode() ^ Name.GetHashCode() ^ //etc.
}

如果您无法修改Person,那么您可以定义自己的IEqualityComparer<Person>并将其移至SequenceEquals,以便它可以执行除默认引用相等之外的相等检查。

更新:如果订单不重要,那么您可以使用UnionExcept。这可能会非常快,因此您至少应该考虑将之前比较的集合转换为某种类型的Set

答案 1 :(得分:1)

您可以使用ICompareable接口并实现自己的比较功能。您可以参考以下链接https://msdn.microsoft.com/de-de/library/system.icomparable(v=vs.110).aspx