C# - 如何让类型可以比较
IComparable<T>
.NET 里,IComparable<T>是用来作比较的最常用接口。
如果某个类型的实例需要与该类型的其它实例进行比较或者排序的话,那么该类型就可以通过实现IComparable<T>接口来达到此目的。
IComparable<T>只提供了一个方法:
先看一个例子,这里使用了string,因为string实现了该接口:
其结果是:
string是通过按位字母进行比较的,“a”就小于“b”,所以上述str1应该是小于str2的。
而CompareTo方法返回的是int类型,而比较的结果呢,可能有三种情况:
- x == y
- x < y
- x > y
再通过上面的例子,我们可以看出来:
针对x.CompareTo(y),
- 如果 x == y,那么 结果 = 0
- 如果 x < y,那么结果 < 0
- 如果 x > y,那么结果 > 0
我们可以把代码重构一下,提取出一个低级别方法,便于逻辑复用:
顺便提一下,string并没有实现> < == 等等操作符。
int
所有的原始类型都实现了IComparable<T>。
所以使用上面的方法,也可以比较原始数据类型:
当然这些类型也可以使用操作符,例如:
而string没有实现这些操作符,所以这样写就是错误的:
相等性 vs 比较
直接看图:
其中,针对比较性,System.object并没有支持,因为对于大多数类型而言,对它们的实例进行比较排序是没有意义的。
例如3 < 4,这样就是合理的;而提交按钮 < 取消按钮,这就没有意义了;这个委托 < 另一个委托,这也没有意义。
针对相等性而言,IEquatable<T>仅仅就是对object里的那些Equals方法的补充。而针对比较性而言,IComparable<T>是主打的方式。
其它的方式都有对应。
下面两个黄色的通过”插件的方式“实现的,这里只提一下,不介绍了。
比较性 只比较值
判断相等性的时候,可能判断的是引用相等或者是值相等。
而进行比较排序的时候,其比较的只能是值,因为对引用进行比较排序是没有意义的。
而==和!=操作符可以为原始数据类型和引用类型来使用,而>, <, >=, <= 只能用于原始数据类型。
在自定义类型上实现比较
其实我通常不在我的类型上去实现IComparable<T>,包括引用类型和原始类型。
因为是这样的,比如说有一个Person(人)这个类型,我想对它排序,按照年龄排序,可以;按照姓名排序,也可以;按照身高排序,也可以;但是没有任何一种排序对人来说是最理所当然的。
更好的办法是实现某种比较器。
但是有时候还是需要实现IComparable<T>,那么下面就讲一下怎么做。
值类型
Person Struct:
如果直接使用我们之前的方法,则会报错:
因为它没实现IComparable<T>接口。
使用大于号小于号的话,也会报错:
因为这个类型也没有实现比较操作符。
实现IComparable<T>接口
很简单,直接调用了字段Height的CompareTo方法,因为int类型实现了IComparable<T>接口。
实现比较操作符
一共四个操作符:<, >, <=, >=,必须都得实现。
代码是:
这个很简单就不解释了。
现在代码不会报错了:
其运行结果是:
运行OK了,看似没问题,然后,还有一个问题:
使用等号判断相等性的代码会报错。
如果你不是用==操作符的话,那么代码是没问题的,也是可以进行比较的,也没人强制要求实现==和!=操作符。但是这很奇怪!因为你说 p1 > p2,这个成立,然后再说 p1 != p2这个就编译错误,那就不合理了。
所以,如果你实现了比较操作符,那么相等性操作符也应该一同实现了:
那么既然==和!=都实现了,那么其它的相等性判断方法也应该一同实现:
- object.Equals()
- object.GetHashCode()
- IEquatable<T>
看起来挺麻烦,但这只是一个struct,还是相对简单的。。。。
但针对struct,其实还没完,还有一个非泛型的IComparable接口,泛型出现之前,一直都是用这个接口的。
这个接口现在来说没什么用了,但是如果有其它遗留的老代码需要使用你这个struct,你可能还需要把这个接口实现一下。。。
C# - 如何让类型可以比较的更多相关文章
- 【.net 深呼吸】细说CodeDom(5):类型成员
前文中,老周已经厚着脸皮介绍了类型的声明,类型里面包含的自然就是类型成员了,故,顺着这个思路,今天咱们就了解一下如何向类型添加成员. 咱们都知道,常见的类型成员,比如字段.属性.方法.事件.表示代码成 ...
- 【.net 深呼吸】细说CodeDom(4):类型定义
上一篇文章中说了命名空间,你猜猜接下来该说啥.是了,命名空间下面就是类型,知道了如何生成命名空间的定义代码,之后就该学会如何声明类型了. CLR的类型通常有这么几种:类.接口.结构.枚举.委托.是这么 ...
- opencv中Mat与IplImage,CVMat类型之间转换
opencv中对图像的处理是最基本的操作,一般的图像类型为IplImage类型,但是当我们对图像进行处理的时候,多数都是对像素矩阵进行处理,所以这三个类型之间的转换会对我们的工作带来便利. Mat类型 ...
- [C#] async 的三大返回类型
async 的三大返回类型 序 博主简单数了下自己发布过的异步文章,已经断断续续 8 篇了,这次我想以 async 的返回类型为例,单独谈谈. 异步方法具有三个可让开发人员选择的返回类型:Task&l ...
- C# - 值类型、引用类型&走出误区,容易错误的说法
1. 值类型与引用类型小总结 1)对于引用类型的表达式(如一个变量),它的值是一个引用,而非对象. 2)引用就像URL,是允许你访问真实信息的一小片数据. 3)对于值类型的表达式,它的值是实际的数据. ...
- salesforce 零基础学习(六十二)获取sObject中类型为Picklist的field values(含record type)
本篇引用以下三个链接: http://www.tgerm.com/2012/01/recordtype-specific-picklist-values.html?m=1 https://github ...
- Dapper逆天入门~强类型,动态类型,多映射,多返回值,增删改查+存储过程+事物案例演示
Dapper的牛逼就不扯蛋了,答应群友做个入门Demo的,现有园友需要,那么公开分享一下: 完整Demo:http://pan.baidu.com/s/1i3TcEzj 注 意 事 项:http:// ...
- ElasticSearch 5学习(9)——映射和分析(string类型废弃)
在ElasticSearch中,存入文档的内容类似于传统数据每个字段一样,都会有一个指定的属性,为了能够把日期字段处理成日期,把数字字段处理成数字,把字符串字段处理成字符串值,Elasticsearc ...
- js:给定两个数组,如何判断他们的相对应下标的元素类型是一样的
题目: 给Array对象原型上添加一个sameStructureAs方法,该方法接收一个任意类型的参数,要求返回当前数组与传入参数数组(假定是)相对应下标的元素类型是否一致. 假设已经写好了Array ...
- C++11特性——变量部分(using类型别名、constexpr常量表达式、auto类型推断、nullptr空指针等)
#include <iostream> using namespace std; int main() { using cullptr = const unsigned long long ...
随机推荐
- springMVC中添加restful 风格
RESTful架构:是一种设计的风格,并不是标准,只是提供了一组设计原则和约束条件,也是目前比较流行的一种互联网软件架构.它结构清晰.符合标准.易于理解.扩展方便,所以正得到越来越多网站的采用. 关于 ...
- Python中urllib.urlencode中文字符的一个问题
Django项目在访问Restful service时调用urllib.urlencode编码中文字符串时碰到下面这个错误. v = quote_plus(str(v)) UnicodeEncodeE ...
- C# 获取文件下载的各种方法
public class RemoteDownload { public static void DownLoad(string addressUrl,string localName) { //下载 ...
- UE4学习心得:Scene Component蓝图的一个简单应用
Scene Component是蓝图类中一个不怎么常用的分类(特别是对于新手而言),主要是其实现的功能可以在Actor类中用相同的方法实现,使其作用显得有点多余. 笔者在使用过这个类之后发现其作用更相 ...
- JQuery制作基础的无缝轮播与左右点击效果
在网页中我们想要的无缝轮播左右循环有好多好多中,这是我第一个轮播效果,也是最基础的,和大家分享一下,对于初学者希望你们能有所借鉴,对于大神我想让你们尽情的虐我给我宝贵的意见. 这个是我要的效果 进入正 ...
- Eclipse开发前,常用设置
设置工作空间的项目编码, 防止出现乱码 Window - Preferences - General - Workspace 将"Text file encoding" ...
- Spring中的循环依赖
循环依赖 在使用Spring时,如果主要采用基于构造器的依赖注入方式,则可能会遇到循环依赖的情况,简而言之就是Bean A的构造器依赖于Bean B,Bean B的构造器又依赖于Bean A.在这种情 ...
- 敏捷(Agile)——“说三道四”
可以这么理解:一种以人为本.团队合作.快速响应变化和可工作的软件作为宗旨的开发方法.亦可理解为在一个高度协作的环境中,不断地使用反馈进行自我调整和完善,持续交付用户想要的软件的过程.敏捷开发提倡通过多 ...
- springMVC(spring)+WebSocket案例(获取请求参数)
开发环境(最低版本):spring 4.0+java7+tomcat7.0.47+sockjs 前端页面要引入: <script src="http://cdn.jsdelivr.ne ...
- Centos下的apache2练习
前言: 我上星期一直在写代码忘记写博客了,明天回去补回来.脚本主要用于收集信息 今天刚刚学完apache.来做个总结,写的不好请多多指指出. 目标: Centos6.5的IP:192.168.1.21 ...