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. 使用GridFsTemplate在mongodb中存取文件

    spring-data-mongodb之gridfs   mongodb除了能够存储大量的数据外,还内置了一个非常好用的文件系统.基于mongodb集群的优势,GridFS当然也是分布式的,而且备份也 ...

  2. 第七章 : Git 介绍 (下)[Learn Android Studio 汉化教程]

    Learn Android Studio 汉化教程 Let’s reset even further to remove all traces of your work on the deprecat ...

  3. springcloud(七) feign + Hystrix 整合 、

    之前几章演示的熔断,降级 都是 RestTemplate + Ribbon 和 RestTemplate + Hystrix  ,但是在实际开发并不是这样,实际开发中都是 Feign 远程接口调用. ...

  4. Vim中异步语法检查ale配置

    注意 在设置let g:ale_sign_error = '✗'和let g:ale_sign_warning = '⚡'这些时,可能vim不让你保存,提示fenc这个东西. 所以,为了保险起见,你最 ...

  5. 2.spring整合ehcache 注解实现查询缓存,并实现实时缓存更新或删除

    转自:http://www.importnew.com/23358.html 写在前面:上一篇博客写了spring cache和ehcache的基本介绍,个人建议先把这些最基本的知识了解了才能对今天主 ...

  6. 少走弯路,给Java 1~5 年程序员的建议

    参考:https://www.jianshu.com/p/5681a1f0aad6 今天LZ是打算来点干货,因此咱们就不说一些学习方法和技巧了,直接来谈每个阶段要学习的内容甚至是一些书籍.这一部分的内 ...

  7. django一对一数据库建立和进行数据传输的3种方式all()(对象) values()(字典) values_list()(元组)

    class Business(models.Model): caption = models.CharField(max_length=32) code = models.CharField(max_ ...

  8. 【317】python 指定浏览器打开网页 / 文件

    一.python 打开浏览器的方法: 1. startfile方法(打开指定浏览器) import os os.startfile("C:\Program Files\internet ex ...

  9. 创建maven版的java工程

    步骤如下: 1.第一步 2.第二步: 3.第三步:

  10. python初步要点

    [python初步要点] #! 用于告诉操作系统去哪里找Python解释器为运行您的程序. 1.print 的输出有以下2种形式,""%()的形式类似于C的printf. 要注意逗 ...