C# - 如何让类型可以比较

简介: C# - 如何让类型可以比较IComparable.NET 里,IComparable是用来作比较的最常用接口。如果某个类型的实例需要与该类型的其它实例进行比较或者排序的话,那么该类型就可以通过实现IComparable接口来达到此目的。

C# - 如何让类型可以比较
IComparable
.NET 里,IComparable是用来作比较的最常用接口。

如果某个类型的实例需要与该类型的其它实例进行比较或者排序的话,那么该类型就可以通过实现IComparable接口来达到此目的。

IComparable只提供了一个方法:

先看一个例子,这里使用了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。

所以使用上面的方法,也可以比较原始数据类型:

当然这些类型也可以使用操作符,例如:

而string没有实现这些操作符,所以这样写就是错误的:

相等性 vs 比较
直接看图:

其中,针对比较性,System.object并没有支持,因为对于大多数类型而言,对它们的实例进行比较排序是没有意义的。

例如3 < 4,这样就是合理的;而提交按钮 < 取消按钮,这就没有意义了;这个委托 < 另一个委托,这也没有意义。

针对相等性而言,IEquatable仅仅就是对object里的那些Equals方法的补充。而针对比较性而言,IComparable是主打的方式。

其它的方式都有对应。

下面两个黄色的通过”插件的方式“实现的,这里只提一下,不介绍了。

比较性 只比较值

判断相等性的时候,可能判断的是引用相等或者是值相等。

而进行比较排序的时候,其比较的只能是值,因为对引用进行比较排序是没有意义的。

而==和!=操作符可以为原始数据类型和引用类型来使用,而>, <, >=, <= 只能用于原始数据类型。

在自定义类型上实现比较
其实我通常不在我的类型上去实现IComparable,包括引用类型和原始类型。

因为是这样的,比如说有一个Person(人)这个类型,我想对它排序,按照年龄排序,可以;按照姓名排序,也可以;按照身高排序,也可以;但是没有任何一种排序对人来说是最理所当然的。

更好的办法是实现某种比较器。

但是有时候还是需要实现IComparable,那么下面就讲一下怎么做。

值类型
Person Struct:

如果直接使用我们之前的方法,则会报错:

因为它没实现IComparable接口。

使用大于号小于号的话,也会报错:

因为这个类型也没有实现比较操作符。

实现IComparable接口

很简单,直接调用了字段Height的CompareTo方法,因为int类型实现了IComparable接口。

实现比较操作符
一共四个操作符:<, >, <=, >=,必须都得实现。

代码是:

这个很简单就不解释了。

现在代码不会报错了:

其运行结果是:

运行OK了,看似没问题,然后,还有一个问题:

使用等号判断相等性的代码会报错。

如果你不是用==操作符的话,那么代码是没问题的,也是可以进行比较的,也没人强制要求实现==和!=操作符。但是这很奇怪!因为你说 p1 > p2,这个成立,然后再说 p1 != p2这个就编译错误,那就不合理了。

所以,如果你实现了比较操作符,那么相等性操作符也应该一同实现了:

那么既然==和!=都实现了,那么其它的相等性判断方法也应该一同实现:

object.Equals()
object.GetHashCode()
IEquatable
看起来挺麻烦,但这只是一个struct,还是相对简单的。。。。

但针对struct,其实还没完,还有一个非泛型的IComparable接口,泛型出现之前,一直都是用这个接口的。

这个接口现在来说没什么用了,但是如果有其它遗留的老代码需要使用你这个struct,你可能还需要把这个接口实现一下。。。

引用类型
引用类型除了需要考虑上面struct考虑的那些东西外,还需要考虑更多的东西。

首先,需要在CompareTo里面检查是否为null,和类型检查。

而如果Person是一个没有seal的class,那问题就更大了,以前文章里提到的OOP继承问题、类型安全问题、相等性问题将全部出现。因为类型安全和比较性还是没法一起很愉快的工作。反正会很混乱。。。

所以如果事seal的class,那么在其上实现比较性的话还勉强可以接受;否则的话,祝好运。。。

泛型
之前在相等性的文章里,提到过,针对泛型代码来说,==和!=操作符不能很好的工作,而object.Equals()却可以。

这点在比较性里面也是一样的。针对泛型的比较,你需要使用IComparable.CompareTo()方法,而不是比较的操作符>, <, >=, <=等(即使实现了比较操作符)。

如果我把之前的方法代码改成使用比较操作符:

那么就会报错,因为无法约束泛型实现了某些操作符。。。但可以考虑在接口里面实现比较操作符。。。

但是实现比较性的话:

实现IComparable接口
也可选去实现比较操作符。
原文地址https://www.cnblogs.com/cgzl/p/10777541.html

相关文章
|
1月前
|
存储 安全 编译器
C# 11.0中的泛型属性:类型安全的新篇章
【1月更文挑战第23天】C# 11.0引入了泛型属性的概念,这一新特性为开发者提供了更高级别的类型安全性和灵活性。本文将详细探讨C# 11.0中泛型属性的工作原理、使用场景以及它们对现有编程模式的改进。通过深入了解泛型属性,开发者将能够编写更加健壮、可维护的代码,并充分利用C#语言的最新发展。
|
9月前
|
编译器 C#
C#之十七 局部类型
C#之十七 局部类型
21 0
|
29天前
|
安全 API C#
C#.Net筑基-类型系统②常见类型--枚举Enum
枚举(enum)是C#中的一种值类型,用于创建一组命名的整数常量。它们基于整数类型(如int、byte等),默认为int。枚举成员可指定值,未指定则从0开始自动递增。默认值为0。枚举可以与整数类型互相转换,并可通过`[Flags]`特性表示位域,支持位操作,用于多选场景。`System.Enum`类提供了如`HasFlag`、`GetName`等方法进行枚举操作。
|
29天前
|
编译器 C#
C#.Net筑基-类型系统②常见类型 --record是什么类型?
`record`在C#中是一种创建简单、只读数据结构的方式,常用于轻量级数据传输。它本质上是类(默认)或结构体的快捷形式,包含自动生成的属性、`Equals`、`ToString`、解构赋值等方法。记录类型可以继承其他record或接口,但不继承普通类。支持使用`with`语句创建副本。例如,`public record User(string Name, int Age)`会被编译为包含属性、相等比较和`ToString()`等方法的类。记录类型提供了解构赋值和自定义实现,如密封的`sealed`记录,防止子类重写。
|
29天前
|
存储 C#
C#.Net筑基-类型系统②常见类型--结构体类型Struct
本文介绍了C#中的结构体(struct)是一种用户自定义的值类型,适用于定义简单数据结构。结构体可以有构造函数,能定义字段、属性和方法,但不能有终结器或继承其他类。它们在栈上分配,参数传递为值传递,但在类成员或包含引用类型字段时例外。文章还提到了`readonly struct`和`ref struct`,前者要求所有字段为只读,后者强制结构体存储在栈上,适用于高性能场景,如Span和ReadOnlySpan。
|
29天前
|
存储 安全 Unix
C#.Net筑基-类型系统②常见类型--日期和时间的故事
在System命名空间中,有几种表示日期时间的不可变结构体(Struct):DateTime、DateTimeOffset、TimeSpan、DateOnly和TimeOnly。DateTime包含当前本地或UTC时间,以及最小和最大值;DateTimeOffset增加了时区偏移信息,适合跨时区操作。UTC是世界标准时间,而格林尼治标准时间(GMT)不稳定,已被更精确的UTC取代。DateTimeOffset和DateTime提供了转换为UTC和本地时间的方法,以及各种解析和格式化函数。
|
1月前
|
存储 C# 开发者
C#变量类型
C#变量类型
29 0
|
1月前
|
开发框架 .NET 编译器
C# 9.0中的目标类型新表达式:类型推断的又一进步
【1月更文挑战第16天】C# 9.0引入了目标类型新表达式,这是类型推断功能的一个重要扩展。通过目标类型新表达式,开发者在创建对象时可以省略类型名称,编译器会根据上下文自动推断所需类型。这一特性不仅简化了代码编写,还提高了代码的可读性和维护性。本文将详细介绍目标类型新表达式的语法、使用场景及其对C#编程的影响。
|
1月前
|
存储 C# 容器
掌握 C# 变量:在代码中声明、初始化和使用不同类型的综合指南
变量是用于存储数据值的容器。 在 C# 中,有不同类型的变量(用不同的关键字定义),例如: int - 存储整数(没有小数点的整数),如 123 或 -123 double - 存储浮点数,有小数点,如 19.99 或 -19.99 char - 存储单个字符,如 'a' 或 'B'。Char 值用单引号括起来 string - 存储文本,如 "Hello World"。String 值用双引号括起来 bool - 存储具有两个状态的值:true 或 false
56 2
|
1月前
|
存储 安全 算法
C# 泛型:类型参数化的强大工具
【1月更文挑战第7天】本文将深入探讨C#语言中的泛型编程,包括泛型的定义、用途、优势以及实际应用。通过类型参数化,泛型允许开发者编写更加灵活且可重用的代码,同时提高程序的类型安全性和性能。本文将通过示例代码和详细解释,帮助读者更好地理解泛型在C#中的重要性和实用性。