如何检查对象内容平等而不是对象引用平等

时间:2012-05-01 15:56:18

标签: c# reflection properties

我有一个实体框架实体,看起来像这样:

class ListItemEtlObject
{
    public int ID { get; set; }
    public string ProjectName { get; set; }
    public string ProjectType { get; set; }
    public string ProjectCode { get; set; }
    public string ProjectDescription { get; set; }
    public string JobNo { get; set; }
    public string JobDescription { get; set; }
    public bool Include { get; set; }
}

我将两个不同数据源中的项目拖入IEnumerable列表中。如何在不使用一堆if语句的情况下比较这些项目,以检查属性值之间是否存在差异,然后设置属性值如果不匹配?我们的想法是保持列表同步。列表A也有ID值设置,列表B没有。我觉得有一个更好的方法来做这个比一堆

if(objectA.ProjectName != objectB.ProjectName)
{
   objectA.ProjectName = objectB.ProjectName;
}

3 个答案:

答案 0 :(得分:3)

如果您拥有源对象的控制权,那么支持基于值的相等性的最佳声明方式是实现IEquatable<T>。遗憾的是,这需要您枚举所有这些检查,但它在实际的对象定义位置完成一次,并且不需要在整个代码库中重复。

class ListItemEtlObject : IEquatable<ListITemEtlObject>
{
  ...
  public void Equals(ListITemEtlObject other) {
    if (other == null) {
      return false;
    }
    return 
      ID == other.ID &&
      ProjectName == other.ProjectName &&
      ProjectType == other.ProjectType &&
      ... ;
  }
}

此外,您可以选择在对象类型上重载相等运算符,并允许使用者在!=实例上简单地使用==ListItemEtlObject,并获得值相等而不是引用相等。

public static bool operator==(ListItemEtlObject left, ListItemEtlObject right) {
  return EqualityComparer<ListItemEtlObject>.Default.Equals(left, right);
}
public static bool operator!=(ListItemEtlObject left, ListItemEtlObject right) {
  return !(left == right);
}

答案 1 :(得分:2)

最简单的方法是在类上提供一个计算特定哈希的方法,就像GetHashCode一样,然后如果两个实例计算相同的哈希值,则可以说它们是等价的。

答案 2 :(得分:1)

您可以使用reflection =)

来简化它
    public virtual void SetDifferences(MyBaseClass compareTo)
    {
        var differences = this.GetDifferentProperties(compareTo);

        differences.ToList().ForEach(x =>
        {
            x.SetValue(this, x.GetValue(compareTo, null), null);
        });
    }

    public virtual IEnumerable<PropertyInfo> GetDifferentProperties(MyBaseClass compareTo)
    {
        var signatureProperties = this.GetType().GetProperties();

        return (from property in signatureProperties
                let valueOfThisObject = property.GetValue(this, null)
                let valueToCompareTo = property.GetValue(compareTo, null)
                where valueOfThisObject != null || valueToCompareTo != null
                where (valueOfThisObject == null ^ valueToCompareTo == null) || (!valueOfThisObject.Equals(valueToCompareTo))
                select property);
    }

以下是我为你做过的几项测试

    [TestMethod]
    public void CheckDifferences()
    {
        var f = new OverridingGetHashCode();
        var g = new OverridingGetHashCode();

        f.GetDifferentProperties(g).Should().NotBeNull().And.BeEmpty();

        f.Include = true;
        f.GetDifferentProperties(g).Should().NotBeNull().And.HaveCount(1).And.Contain(f.GetType().GetProperty("Include"));

        g.Include = true;
        f.GetDifferentProperties(g).Should().NotBeNull().And.BeEmpty();

        g.JobDescription = "my job";
        f.GetDifferentProperties(g).Should().NotBeNull().And.HaveCount(1).And.Contain(f.GetType().GetProperty("JobDescription"));
    }

    [TestMethod]
    public void SetDifferences()
    {
        var f = new OverridingGetHashCode();
        var g = new OverridingGetHashCode();

        g.Include = true;
        f.SetDifferences(g);
        f.GetDifferentProperties(g).Should().NotBeNull().And.BeEmpty();

        f.Include = true;
        g.Include = false;
        f.SetDifferences(g);
        f.GetDifferentProperties(g).Should().NotBeNull().And.BeEmpty();
        f.Include.Should().BeFalse();
    }