C#实现类型的比较示例详解
作者:solenovex 发布时间:2022-02-18 18:17:44
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,你可能还需要把这个接口实现一下。。。😂
引用类型
引用类型除了需要考虑上面struct考虑的那些东西外,还需要考虑更多的东西。
首先,需要在CompareTo里面检查是否为null,和类型检查。
而如果Person是一个没有seal的class,那问题就更大了,以前文章里提到的OOP继承问题、类型安全问题、相等性问题将全部出现。因为类型安全和比较性还是没法一起很愉快的工作。反正会很混乱。。。
所以如果事seal的class,那么在其上实现比较性的话还勉强可以接受;否则的话,祝好运。。。
泛型
之前在相等性的文章里,提到过,针对泛型代码来说,==和!=操作符不能很好的工作,而object.Equals()却可以。
这点在比较性里面也是一样的。针对泛型的比较,你需要使用IComparable<T>.CompareTo()方法,而不是比较的操作符>, <, >=, <=等(即使实现了比较操作符)。
如果我把之前的方法代码改成使用比较操作符:
那么就会报错,因为无法约束泛型实现了某些操作符。。。但可以考虑在接口里面实现比较操作符。。。
但是实现比较性的话:
实现IComparable<T>接口
也可选去实现比较操作符。
来源:http://www.cnblogs.com/cgzl/p/10777541.html
猜你喜欢
- 代理模式是java中最常用的设计模式之一,尤其是在spring框架中广泛应用。对于java的代理模式,一般可分为:静态代理、 * 、以及C
- 枚举类型是一种的值类型,它用于声明一组命名的常数。(1)枚举的声明:枚举声明用于声明新的枚举类型。访问修辞符 enum 枚举名:基础类型&n
- 一、定时任务的使用场景和常见的定时任务某个时间定时处理某个任务、发邮件、短信、消息提醒、订单通知、统计报表等定时任务划分单机定时任务:单机的
- 首先,将json串转为一个JObject对象:JObject jo = (JObject)JsonConvert.DeserializeOb
- 前言:在java的网络通信中,两个不同节点的主机想要进行通信则可以通过建立Socket对象(相当于客户端主机,向服务端请求发送信息)和Ser
- 前些天在写个小程序,用到DataGridView,想给它动态的显示行号。不是很费劲GOOGLE了一下,这GOOGLE不要紧,发现了不少问题。
- 之前文章中我们讲到,java中实现同步的方式是使用synchronized block。在java 5中,Locks被引入了,来提供更加灵活
- 本文实例讲述了C#中static静态变量的用法。分享给大家供大家参考。具体如下:使用 static 修饰符声明属于类型本身而不是属于特定对象
- 这篇文章主要介绍了Java数组索引异常产生及解决方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友
- java 实现MD5加密算法的简单实例实现代码:import java.security.NoSuchAlgorithmException;
- 这两个类使用起来非常方便,可以完成我们对定时器的绝大多数需求Timer类是用来执行任务的类,它接受一个TimerTask做参数Timer有两
- 一、前言1、热更新代码的场景(1)当线上服务器出现问题时,有些时候现有的手段不足以发现问题所在,可能需要追加打印日志或者增加一些调试代码,如
- 本文实例为大家分享了Java实现五子棋网络版的具体代码,供大家参考,具体内容如下需求分析:对于网络五子棋而言,在普通五子棋的基础上需要添加以
- 这篇文章主要介绍了Spring AOP AspectJ使用及配置过程解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考
- 一棵二叉查找树是按二叉树结构来组织的。这样的树可以用链表结构表示,其中每一个结点都是一个对象。结点中除了数据外,还包括域left,right
- 首先需要在 AndroidManifest.xml 文件中添加「获取模拟定位信息」权限。<uses-permission androi
- -----------------------------------------------------操作---------------
- 本文实例讲述了Java日期操作方法工具类。分享给大家供大家参考,具体如下:package com.gcloud.common;import
- Condition的作用是对锁进行更精确的控制。Condition中的await()方法相当于Object的wait()方法,Conditi
- 部署到webapps目录启动本文使用的Spring版本为Spring6,SpringBoot版本为3,JDK为17,可能会和之前有细微不同,