C# - 为值类型重定义相等性
C# - 为值类型重定义相等性
为什么要为值类型重定义相等性
原因主要有以下几点:
值类型默认无法使用 == 操作符,除非对它进行重写
再就是性能原因,因为值类型默认的相等性比较会使用装箱和反射,所以性能很差
根据业务需求,其实际相等性的意义和默认的比较结果可能会不同,但是这种情况可能不较少
所以建议是:所有供外部使用的struct都实现相等性。
实现步骤
重写object.Equals()方法
实现IEquatable.Equals()接口方法
重写 == 和 != 操作符
重写object.GetHashCode()
具体来说:
重写object.Equals()方法,是避免了反射,因为System.ValueType里面对object.Equals()方法的重写实现如下:
这里用到了反射。
而实现IEquatable.Equals()接口方法,可以避免装箱,并且保证类型安全。
而实现==和!=,也就允许值类型使用该操作符了,写起来更方便直观,易于理解。而且这两个操作符必须一同实现。
而重写object.GetHashCode(),则是一个最佳实践。
所有为值类型重定义相等性,一共分4步,每步都是必须的。
实现
先看实例struct:
有构造函数,涉及到一个enum,并重写了ToString()方法。
实现IEquatable接口
首先来实现IEquatable接口。
(如果你使用resharper或者Rider,那么实现该接口的时候它会自动把object的Equals和GetHashCode方法都重写了,并且自动完成了有意义的代码)
这里面我对三个属性进行了比较,使用了==操作符。其中==对于string来说就是比较值,而enum其实就是int,DateTime也是值类型,并且已经实现了相等性判断的功能。
重写object.Equals()方法
这个代码是resharper生成的。
代码很简单,首先检查是否为null,然后检查这个object是不是一个Person,这里使用了 is 操作符,并把它转型为Person,赋给了一个叫做other的变量。最后调用的这个Equals()方法,是我们上面写的那个强类型的方法,因为other变量的类型是Person。
但是这个方法仍然涉及到装箱操作,所以还是IEquatable的实现方法更快一些。
如果只重写了object.Equals()方法,而没有重写GetHasCode()方法,那么resharper会有提示:
实现 == 和 != 操作符
这个很简单,直接调用强类型的Equals()方法即可,而且由于Person是值类型,所以不用检查null,值类型不会为null。
如果只实现了其中一个操作符,那么会报错的。
实现object.GetHashCode()
GetHashCode()这个方法会返回一个32位的哈希码,它代表着对象内容的哈希值。
而类型里拥有GetHashCode()方法(返回Hash)的真正目的是,允许该类型在内部使用HashTable的集合中可以作为Key,因为HashTable需要这些哈希码。例如Dictionary。
为了让HashTable可以正确的工作,Hash码有一个要求:如果两个实例被认为是相等的,那么它们必须返回相同的hash码。如果没有实现这个要求,那么你可能会发现这个类型作为Dictionary的Key的时候,会有一些意想不到的结果。
所以如果重写了object.Equals()方法,那么就得重写object.GetHashCode()方法。
看一下resharper自动实现的代码:
这里使用了unchecked,防止抛出溢出异常。
Name是引用类型,可能为null,所以判断一下。
然后其它两个int和DateTime类型,微软都做好了其GetHashCode()的实现。
这里对它们进行异或操作。之所以使用397这个数,可能因为397是一个足够大的质数,可以导致溢出,并混淆各位,之所以使用质数,是因为用质数相乘会得到比用其他任意数相乘更均匀的结果。
检验
结果如预期,OK。
总结
在这几个动作里,实际的逻辑写在了IEquatable.Equals()方法里,object.Equals()就是检查类型然后调用IEquatable.Equals(),== 和 != 操作符也是调用IEquatable.Equals(),而GetHashCode()则使用了按位异或。
最后再重复一次,为值类型定义相等性一定要实现上述4各步骤的5个方法。
原文地址https://www.cnblogs.com/cgzl/p/10699667.html
低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
java web 项目中封装的下拉列表小组件:实现下拉列表使用者前后端0行代码
导读: 主要从4个方面来阐述,1:背景;2:思路;3:代码实现;4:使用 一:封装背景 像easy ui 之类的纯前端组件,也有下拉列表组件,但是使用的时候,每个下拉列表,要配一个URL ,以及设置URL反回来的值和 select 的text ,和value 的对应关系 ,这有2个问题:一使用者必须知道URL ,二,如果页面有10个下拉表表,要请求后台10次,肯定影响性能,而我想要的是使用者只要申明用哪个数据字典就行了,其他根本不用操心,另外加上在做itest开测试测试管理项目的时候,有几个页面,特别多下拉列表,且是动态数据,到处都有处理下拉表列表,后台代码还好,前端到处都要用JS处理,就算是用vue ,或理angular JS 一样要处理,我这人又很懒, 最怕重复的代码,千女散花似的散落在各个角落中,一不做,二不休干脆不如简单的写一个组件(前后端都有的),让使用者前后端0行代码。我们先来看看一下,itest 开源测试管理项目中这个界面,下拉列表,多得头大,处理不好,会很慢。可以在这体验这个多下拉列表页面(点测试,然后选择一个项目,然后点缺陷管理,再点增加),体验地址:https:/...
- 下一篇
linux服务器上使用find查杀webshell木马方法
linux服务器上使用find查杀webshell木马方法本文转自:http://ju.outofmemory.cn/entry/256317 只要从事互联网web开发的,都会碰上web站点被入侵的情况。这里我把查杀的一些方法采用随记的形式记录一下,一是方便自己以后的工作需要,二是给其他朋友提供一些参考帮助。写的不周的地方,高手们不要喷,欢迎给我提供更好的意见,对于我个人来说也是个提高,并表示感谢。 我们的服务器环境是linux,所以,肯定少不了用find这个命令,并且需要配合ls命令来使用。 1、可以查找近3天被修改过的文件,并显示文件列表详细信息: find -name "*.php" -type f -mtime -3 -exec ls -l {} ;当然,结果中可能会包含很多cache类文件,这些文件不是我们要查找的,那么就需要把这类文件从查询结果中排除掉,往往cache文件都存放到cache特定的目录。 使用 -prune 参数来进行过滤,增加排除某些目录条件的查询命令: find . -path "/xxxxx/caches" -prune -o -name "*.php"...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- SpringBoot2初体验,简单认识spring boot2并且搭建基础工程
- CentOS7,CentOS8安装Elasticsearch6.8.6
- SpringBoot2整合MyBatis,连接MySql数据库做增删改查操作
- CentOS7,8上快速安装Gitea,搭建Git服务器
- Springboot2将连接池hikari替换为druid,体验最强大的数据库连接池
- MySQL8.0.19开启GTID主从同步CentOS8
- Docker使用Oracle官方镜像安装(12C,18C,19C)
- SpringBoot2全家桶,快速入门学习开发网站教程
- CentOS6,7,8上安装Nginx,支持https2.0的开启
- CentOS8编译安装MySQL8.0.19