背景

最近在群里,有人问如何深度比较2个对象相等,感觉很有意思,就自己研究了一下,并写了一个开源的小类库,地址如下https://github.com/lamondlu/ObjectEquality。

如果想直接使用这个类库,可以使用Nuget进行安装

Install-Package ObjectEquality

对象比较有几种情况

  1. 对象是值类型或者String,这里仅需要判断值是否相等
  2. 对象是Struct,需要判断Struct的每个字段是否一致
  3. 对象是集合,需要判断对应位置的对象是否相等
  4. 对象是数组,需要判断对应位置的对象是否相等
  5. 对象是Class, 需要判断Class的每个字段是否一致

这里可能有缺漏,大家可以帮我补充。

编写代码

这里我首先创建了一个IEquality接口,在其中定义了一个IsEqual方法,这个方法就是判断2个对象是否一致的方法。后面我会针对上面说明的几种对比场景,分别创建对应的实现类。

    public interface IEquality
{
Func<object, bool> MatchCondition { get; }
bool IsEqual(object source, object target);
}

这里MatchCondition是一个委托,它定义了当前对比类的匹配条件。

第二步,我们针对上述的几种对比场景,创建对应的实现类

值类型相等判断实现类

    internal class ValueTypeEquality : IEquality
{
public Func<object, bool> MatchCondition
{
get
{
return p => p.GetType().IsValueType || p.GetType() == typeof(string);
}
} public bool IsEqual(object source, object target)
{
return source.Equals(target);
}
}

值类型的判断比较简单,直接调用Object类的Equals方法即可。

String类型虽然不是值类型,但是这里我们需要把它归到值类型中。

Struct相等判断实现类

    internal class StructEquality : IEquality
{
public Func<object, bool> MatchCondition
{
get
{
return p => p.GetType().IsValueType
&& !p.GetType().IsPrimitive
&& !p.GetType().IsEnum;
}
} public bool IsEqual(object source, object target)
{
var type = source.GetType(); foreach (var prop in type.GetProperties())
{
var equality = EqualityCollection.Equalities
.First(p => p.MatchCondition(prop.GetValue(source))); var result = equality.IsEqual(prop.GetValue(source),
prop.GetValue(target)); if (!result)
{
return false;
}
} return true;
}
}

这里我们读取了Struct中的每个属性,分别进行判断,如果有一个判断失败,即认为2个Struct对象不相等。

这里EqualityCollection是判断器集合,后续会添加这个类的代码。

集合相等判断实现类

	internal class GenericCollectionEquality : IEquality
{
public Func<object, bool> MatchCondition
{
get
{
return p => p.GetType().IsGenericType;
}
} public bool IsEqual(object source, object target)
{
var type = source.GetType();
var genericType = type.GetGenericArguments()[0]; var genericCollectionType = typeof(IEnumerable<>).MakeGenericType(genericType); if (type.GetInterfaces().Any(p => p == genericCollectionType))
{
var countMethod = type.GetMethod("get_Count"); var sourceCount = (int)countMethod.Invoke(source, null);
var targetCount = (int)countMethod.Invoke(target, null); if (sourceCount != targetCount)
{
return false;
} var sourceCollection = (source as IEnumerable<object>).ToList();
var targetCollection = (target as IEnumerable<object>).ToList(); for (var i = 0; i < sourceCount; i++)
{
var equality = EqualityCollection.Equalities.First(p => p.MatchCondition(sourceCollection[i])); var result = equality.IsEqual(sourceCollection[i], targetCollection[i]); if (!result)
{
return false;
}
}
} return true;
}
}

这里我们首先判断了集合的元素的数量是否一致,如果不一致,即这2个集合不相等。如果一致,我们继续判断对应位置的每个元素是否一致,如果全部都一直,则2个集合相当,否则2个集合不相等。

数组相等判断实现类

	internal class ArrayEquality : IEquality
{
public Func<object, bool> MatchCondition
{
get
{
return p => p.GetType().IsArray;
}
} public bool IsEqual(object source, object target)
{
Array s = source as Array;
Array t = target as Array; if (s.Length != t.Length)
{
return false;
} for (var i = 0; i < s.Length; i++)
{
var equality = EqualityCollection.Equalities
.First(p => p.MatchCondition(s.GetValue(i))); var result = equality.IsEqual(s.GetValue(i), t.GetValue(i)); if (!result)
{
return false;
}
} return true;
}
}

数组相等的判断类似集合,我们首先判断数组的长度是否一致,然后判断对应位置的元素是否一致。

类判断相等实现类

	internal class ClassEquality : IEquality
{
public Func<object, bool> MatchCondition
{
get
{
return p => p.GetType().IsClass;
}
} public bool IsEqual(object source, object target)
{
var type = source.GetType(); foreach (var prop in type.GetProperties())
{
var equality = EqualityCollection.Equalities
.First(p => p.MatchCondition(prop.GetValue(source))); var result = equality.IsEqual(prop.GetValue(source), prop.GetValue(target)); if (!result)
{
return false;
}
} return true;
}
}

添加判断相等实现类集合

	public static class EqualityCollection
{
public static readonly List<IEquality> Equalities = new List<IEquality> {
new StructEquality(),
new ValueTypeEquality(),
new ArrayEquality(),
new GenericCollectionEquality(),
new ClassEquality()
};
}

这里我们定义了一个静态类,来存储程序中使用的所有判断器。

这里在判断器集合中,实现类的其实是有顺序的,StructEquality必须要放到ValueTypeEquality的前面,因为Struct也是值类型,如果不放到最前面,会导致判断失败。

创建判断器入口类

	public class ObjectEquality
{
public bool IsEqual(object source, object target)
{
if (source.GetType() != target.GetType())
{
return false;
} if (source == null && target == null)
{
return true;
}
else if (source == null && target != null)
{
return false;
}
else if (source != null && target == null)
{
return false;
} var equality = EqualityCollection.Equalities
.First(p => p.MatchCondition(source)); return equality.IsEqual(source, target);
}
}

前面所有实现类的访问级别都是Internal, 所以我们需要创建一个判断器入口类, 外部只能通过ObjectEquality类的实例来实现判断相等。

最终效果

下面我列举几个测试用例,看看效果是不是我们想要的

对比Struct

	public struct DemoStruct
{
public int Id { get; set; }
public string Name { get; set; }
}
    var a = new DemoStruct();
a.Id = 1;
a.Name = "Test"; var b = new DemoStruct();
b.Id = 1;
b.Name = "Test"; var c = new DemoStruct();
b.Id = 2;
b.Name = "Test"; ObjectEquality objectEquality = new ObjectEquality();
objectEquality.IsEqual(a,b); //true
objectEquality.IsEqual(a,c); //false

对比类

	public class SimpleClass
{
public int Id { get; set; }
public string Name { get; set; }
}
    var a = new SimpleClass
{
Id = 1,
Name = "A"
}; var b = new SimpleClass
{
Id = 1,
Name = "A"
}; var c = new SimpleClass
{
Id = 2,
Name = "A"
}; ObjectEquality objectEquality = new ObjectEquality();
objectEquality.IsEqual(a,b); //true
objectEquality.IsEqual(a,c); //false

对比数组

 	var a = new int[] { 1, 2, 3 };
var b = new int[] { 1, 2, 3 };
var c = new int[] { 1, 1, 2 }; ObjectEquality objectEquality = new ObjectEquality();
objectEquality.IsEqual(a,b); //true
objectEquality.IsEqual(a,c); //false

.NET中如何深度判断2个对象相等的更多相关文章

  1. python中的is判断引用的对象是否一致,==判断值是否相等

    python中的is判断引用的对象是否一致,==判断值是否相等 a = 10 b = 20 list = [1,2,3,4,5] print(a in list) print(b not in lis ...

  2. JS深度判断两个对象字段相同

    代码: /** * 判断此对象是否是Object类型 * @param {Object} obj */ function isObject(obj){ return Object.prototype. ...

  3. JS深度判断两个数组对象字段相同

    /** * 判断此对象是否是Object类型 * @param {Object} obj */ function isObject(obj){ return Object.prototype.toSt ...

  4. 在ios开发中nil和NUll和Nilde区别————和如何判断连个对象的关系和UISlider不能拖动的问题

    nil表示一个对象指针为空,针对对象 >示例代码: NSString *someString = nil; NSURL *someURL = nil; id someObject = nil; ...

  5. Map.containsKey方法——判断Map集合对象中是否包含指定的键名

    该方法判断Map集合对象中是否包含指定的键名.如果Map集合中包含指定的键名,则返回true,否则返回false. public static void main(String[] args) { M ...

  6. JS中如何判断对象是对象还是数组

    JS中如何判断对象是对象还是数组 一.总结 一句话总结:typeof Array.isArray === "function",Array.isArray(value)和Objec ...

  7. python中判断两个对象是否相等

    #coding=utf-8#比较两个对象是否相等#python 2中使用cmp(),==,is#is 主要是判断 2 个变量是否引用的是同一个对象,如果是的话,则返回 true,否则返回 false. ...

  8. Java中如何判断两个对象是否相等(Java equals and ==)

    原文https://www.dutycode.com/post-140.html 如何判断两个对象相等,这个问题实际上可以看做是如何对equals方法和hashcode方法的理解. 从以下几个点来理解 ...

  9. js判断对象中是否存在某一项和判断是否是对象

    1.判断是否为对象 let str = { name: '第一', age: 12 } console.log(typeof str== "object") 2.判断对象中是否有某 ...

随机推荐

  1. 关于<超文本>定义

    百度百科定义: 1, 超文本是用超链接的方法,将各种不同空间的文字信息组织在一起的网状文本.它更是一种用户界面范式,用以显示文本及与文本之间相关的内容.现时超文本普遍以电子文档方式存在,其中的文字包含 ...

  2. 用ttBulkCp把excel中的数据导入到timesten数据库中

    最近要做数据预处理,需要用到数据库.而且是以前从来没听说过的TimesTen. 首要目标是要把Excel里的数据,导入到TimesTen数据库中.而TimesTen在win10里用不了,于是我就在虚拟 ...

  3. VB.NET或C#报错:You must hava a license to use this ActiveX control.

    VB.NET或者C# winform开发时,如果使用了Microsoft Visual Basic 6.0 ActiveX,并动态创建该控件实例,那么程序移植到没有安装Visual Basic 6.0 ...

  4. Spring AOP Capabilities ang goals

    Spring AOP 是用纯JAVA 实现的. 不需借助JAVA代码在编译处理阶段来实现. Spring 是在运行期实现的.AOP的实现可以在编译,加载,运行三个阶段来实现:Spring AOP 也不 ...

  5. leetcode-求众数

    题目:求众数 给定一个大小为 n 的数组,找到其中的众数.众数是指在数组中出现次数大于 ⌊ n/2 ⌋ 的元素. 你可以假设数组是非空的,并且给定的数组总是存在众数. 示例 1: 输入: [3,2,3 ...

  6. c# 通过MailHelper发送QQ邮件

    发送的方法 appsetting内容 第一个是发送邮件qq账号,第二个是QQ邮箱的POP3/SMTP服务码(下面会说怎么获取),第三个是服务器,第四个是端口 获取QQ邮箱的POP3/SMTP服务码 1 ...

  7. jieba库的使用与词频统计

    1.词频统计 (1)词频分析是对文章中重要词汇出现的次数进行统计与分析,是文本 挖掘的重要手段.它是文献计量学中传统的和具有代表性的一种内容分析方法,基本原理是通过词出现频次多少的变化,来确定热点及其 ...

  8. python os.walk()方法--遍历当前目录的方法

    前记:有个奇妙的想法并想使用代码实现,发现了一个坑,百度了好久也没发现的"填坑"的文章~~~~~~~~~ 那就由我来填 os.walk()支持相对路径 例如 os.walk(&qu ...

  9. js-day02-BOM和DOM

    BOM和Document对象常见属性和方法: BOM是browser object model的缩写,简称浏览器对象模型. Document 对象每个载入浏览器的 HTML 文档都会成为 Docume ...

  10. C++通过GetAdapatersInfo获取网卡配置信息

    DWORD GetAdaptersInfo( PIP_ADAPTER_INFO pAdapterInfo, //指向一个缓冲区,用来取得IP_ADAPTER_INFO结构列表 PULONG pOutB ...