1、引言

  首先我们先来看看IEquatable<T>接口的出现解决了什么问题。

  我们知道,Object基类的Equals方法存在两个明显的问题。一是缺乏类型安全性,二是对于值类型而言需要装箱。在本文中我们就来看下IEquatable<T> Interface是如何解决这两个问题的。

2、IEquatable<T>接口

  我们都知道的一个事实是:如果想让Object的Equals方法为所有派生类型所用,那么,它的参数就必须设计成object类型。

  object是引用类型,这就意味着,如果传递一个值类型的参数,那么该参数将被装箱,这就会造成性能损失。

  另外,还存在另一个问题:将object类型设为参数还意味着类型安全性的缺失。

  解决装箱和类型安全性问题的一个办法就是定义一个新的Equals方法,该方法接受一个和待比较类型相同类型的参数。例如,对于字符串类型而言,定义一个接受string类型的Equals方法就能圆满解决这两个问题。

  但这会面临另一个新的问题,那就是:定义强类型的方法和OOP中的继承存在根本的冲突。我们不能在Object基类中定义一个强类型的Equals方法,因为Object基类根本无法知晓派生类的类型。

  那么,我们怎么样才能定义一个强类型的Equals方法,同时该方法能被所有类型使用呢?微软解决这个问题的思路就是通过提供一个IEquatable<T>接口,该接口向所有类型暴露。查看该接口的定义时,可以发现它仅暴露了一个Equals方法,如下所示。

using System;

namespace System
{
public interface IEquatable<T>
{
bool Equals(T other);
}
}

  该Equals方法和Object基类的虚Equals方法的作用相同,只不过它接受一个T类型参数,因此,它是强类型的,这意味着对于值类型而言,不存在装箱的问题。

3、IEquatable<T>接口和值类型

  我们可以通过一个简单的例子来证明IEquatable<T>接口的使用。

static void Main(String[] args)
{
int number1 = 1;
int number2 = 2;
int number3 = 1; Console.WriteLine(number1.Equals(number2));
Console.WriteLine(number1.Equals(number3)); }

  在上面的例子中,我们定义了三个整型变量,然后使用Equals方法进行比较。在VS中借助智能感知,可以发现对于int类型而言存在两个Equals方法,一个接受object参数,另一个接受int类型参数。接受int参数的Equals方法实现了IEquatable<T>接口,其中,T为int类型。因为我们在调用Equals方法时传递的是一个int类型变量,而不是一个object变量,因此,编译器将选择实现了IEquatable<T>接口的Equals方法。

  在平常开发中对于int类型的比较,我们不会像上面那样使用Equals方法进行比较,而是使用更加简便明了的==操作符。

  所有的基元类型都提供了对IEquatable<T>接口的实现,就像上面代码中的int类型那样,int类型实现了IEquatable<int>。

  总体而言,IEquatable<T>接口对值类型非常有用。但微软并没有为FCL中的非基元的值类型实现该接口,因此,不能寄希望于对FCL中值类型而言总是可以使用该接口。

4、IEquatable<T>和引用类型

  对于引用类型而言,IEquatable<T>接口并没有那么有用。一是因为引用类型不存在像值类型那样的由装箱导致的性能问题,二是因为IEquatable<T>接口不能很好地处理继承问题。

  但值的注意的是,String类型实现了IEquatable<T>接口,如下面所示

static void Main(String[] args)
{
string s1 = "Hello World";
string s2 = string.Copy(s1); Console.WriteLine(s1.Equals(s2); }  

  上面的代码中,C#编译器将直接选择强类型的Equals方法。另外,String类型是sealed的,因此,你不能从它继承。这样,在相等性判定和继承之间的冲突就不存在了。

  很明显,若一个类型定义了两个Equals方法,我们希望它们对相同的输入,产生相同的输出。关于这一点,微软提供的默认实现都严格履行了这一点。当我们自己去实现IEquatable<T>接口时,也要保证这一点。否则,别的开发者使用你定义的类型时将感到困惑。

浅析C#中的IEquatable<T>接口的更多相关文章

  1. 浅析JS中的模块规范(CommonJS,AMD,CMD)////////////////////////zzzzzz

    浅析JS中的模块规范(CommonJS,AMD,CMD)   如果你听过js模块化这个东西,那么你就应该听过或CommonJS或AMD甚至是CMD这些规范咯,我也听过,但之前也真的是听听而已.     ...

  2. 浅析c#中==操作符和equals方法

    在之前的文章中,我们讲到了使用C#中提供的Object类的虚Equals方法来判断Equality,但实际上它还提供了另外一种判断Equality的方法,那就是使用==运算符.许多童鞋也许会想当然的认 ...

  3. 浅析JDK中ServiceLoader的源码

    前提 紧接着上一篇<通过源码浅析JDK中的资源加载>,ServiceLoader是SPI(Service Provider Interface)中的服务类加载的核心类,也就是,这篇文章先介 ...

  4. 通过源码浅析Java中的资源加载

    前提 最近在做一个基础组件项目刚好需要用到JDK中的资源加载,这里说到的资源包括类文件和其他静态资源,刚好需要重新补充一下类加载器和资源加载的相关知识,整理成一篇文章. 理解类的工作原理 这一节主要分 ...

  5. 浅析Java中的深拷贝和浅拷

      浅析Java中的深拷贝和浅拷贝 原文链接: http://blog.csdn.net/tounaobun/article/details/8491392 假如说你想复制一个简单变量.很简单: in ...

  6. 浅析Java中的native关键字

    浅析Java中的native关键字 native关键字说明其修饰的方法是一个原生态方法,方法对应的实现不是在当前文件,而是在用其他语言(如C和C++)实现的文件中.Java语言本身不能对操作系统底层进 ...

  7. c# 把一个匿名对象赋值给一个Object类型的变量后,怎么取这个变量? c# dynamic动态类型和匿名类 详解C# 匿名对象(匿名类型)、var、动态类型 dynamic 深入浅析C#中的var和dynamic

    比如有一个匿名对象,var  result =......Select( a=>new {  id=a.id, name=a.name});然后Object  obj =  result ;我怎 ...

  8. 浅析 Linux 中的时间编程和实现原理一—— Linux 应用层的时间编程【转】

    本文转载自:http://www.cnblogs.com/qingchen1984/p/7007631.html 本篇文章主要介绍了"浅析 Linux 中的时间编程和实现原理一—— Linu ...

  9. 深入浅析python中的多进程、多线程、协程

    深入浅析python中的多进程.多线程.协程 我们都知道计算机是由硬件和软件组成的.硬件中的CPU是计算机的核心,它承担计算机的所有任务. 操作系统是运行在硬件之上的软件,是计算机的管理者,它负责资源 ...

随机推荐

  1. 简单的自动化测试模型(python+selenium)

             刚接触自动化测试,由于没有编程语言的基础,是搞不懂代码里面的函数.封装.包以及其他概念,只是了解字符串.数组.元组及字典这种最基本的名词,更不懂自动化测试框架了.          ...

  2. eclipse怎么自定义工具栏

    1.点击透视图按钮---->右键---->Customize: 2.勾选或者去掉相关项目:

  3. Spring Cloud Bus实现自动更新配置

    一.概述 1. 配置环境 版本:Spring Boot版本2.0.3.RELEASE,Spring Cloud版本Finchley.SR1,RabbitMQ 3.7.7 说明:本文章是在https:/ ...

  4. shell脚本学习指南-学习(1)

    1.先看下面这个命令: $who  | wc  -l  计算当前登陆的用户个数: $who   当前登陆的有哪些用户: pipeling(   |  )可以在两个程序之间建立管道,左侧的结果成为右侧的 ...

  5. 16.3Sum Closest (Two-Pointers)

    Given an array S of n integers, find three integers in S such that the sum is closest to a given num ...

  6. 数据结构notes

    1. 一份很好的数据结构教程,图文并茂,简明扼要,列出每种结构的定义和优缺点,非常适合初学者 via @ranyif https://www.interviewcake.com/data-struct ...

  7. 一个合约访问另一个合约中的mapping

    参考链接:https://ethereum.stackexchange.com/questions/13616/accessing-a-public-mapping-within-a-contract ...

  8. [SoapUI] EndPoint不需要在配置文件中设置不同环境的值,SoapUI自带此参数的设置

    在Environments里面设置就好了

  9. mongo学习-TTL索引 过期数据

    在mongo中我们可以设置文档的过期时间,超过时间,文档会自动删除.(2.x版本中  固定结合也支持,但是到了3.x中 固定集合这个索引不好用) 用法: 1.创建一个db:db.createColle ...

  10. 使用WCF实现消息推送

    1.协议 using System; using System.Collections.Generic; using System.Linq; using System.Text; using Sys ...