欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

C#比较两个对象是否相等(深度比较)

程序员文章站 2022-04-10 13:57:34
...

上篇文章《C#之相等比较(常规比较)》写了C#中比较是否相等的常规方法的使用说明,但是在实际开发中,往往也会用到两个引用类型的对象进行比较,而引用类型中有包含引用类型的属性或字段。
例如: 有一个引用类型TestClass,TestClass中同时包含值类型(int)的属性和引用类型属性(ExClass),引用类型的属性中又包含了引用类型的属性(ExClass2)。

    public class TestClass
    {
        public int ID { get; set; }
        public string Name { get; set; }
        public ExClass OtherCalss { get; set; }
    }

    public class ExClass
    {
        public string Param1 { get; set; }
        public ExClass2 OtherCalss2 { get; set; }
    }

    public class ExClass2
    {
        public string P1 { get; set; }
        public string P2 { get; set; }
    }

这个时候,如果有两个TestClass的实例,c1 ,c2

static void Main(string[] args)
{
    TestClass c1 = new TestClass();
    TestClass c2 = new TestClass();
    //TODO:c1.ID=?   c1.Name=?   c1.OtherClass = ?……
    //TODO:c2.ID=?   c2.Name=?   c2.OtherClass = ?……
    //问题: c1的各属性值是否等于c2的各属性值
}

假设:c1,c2各自经过不同的操作进行了赋值,现在怎么比较c1和c2包含的属性值是否相等???

以上问题是C#提供的常规方法实现不了的(如果有,请各位大牛留言告知,感激不尽……

因此,我就想办法自己写了一个比较的方法,称之为“深度比较”,就是比较c1和c2所有属性值是否相等,包括属性值的属性值

        /// <summary>
        /// 判断两个相同引用类型的对象的属性值是否相等
        /// </summary>
        /// <typeparam name="T"></typeparam>
        /// <param name="obj1">对象1</param>
        /// <param name="obj2">对象2</param>
        /// <param name="type">按type类型中的属性进行比较</param>
        /// <returns></returns>
        public static bool CompareProperties<T>(T obj1, T obj2, Type type)
        {
            //为空判断
            if (obj1 == null && obj2 == null)
                return true;
            else if (obj1 == null || obj2 == null)
                return false;

            Type t = type;
            PropertyInfo[] props = t.GetProperties();
            foreach (var po in props)
            {
                if (IsCanCompare(po.PropertyType))
                {
                    if (!po.GetValue(obj1).Equals(po.GetValue(obj2)))
                    {
                        return false;
                    }
                }
                else
                {
                    return CompareProperties(po.GetValue(obj1), po.GetValue(obj2), po.PropertyType);
                }
            }

            return true;
        }

        /// <summary>
        /// 该类型是否可直接进行值的比较
        /// </summary>
        /// <param name="t"></param>
        /// <returns></returns>
        private static bool IsCanCompare(Type t)
        {
            if (t.IsValueType)
            {
                return true;
            }
            else
            {
                //String是特殊的引用类型,它可以直接进行值的比较
                if (t.FullName == typeof(String).FullName)
                {
                    return true;
                }
                return false;
            }
        }

除了属性值的比较,还可以比较字段值是否相等,原理相同,完整的测试代码在git上有,
地址:https://github.com/PandaCuipp/ObjectCompares

另外:网上有一种方法,是将c1、c2分别 二进制序列化后比较md5值,不知是否可行,还未试验(已验证,代码已更新到上面的git中)