写在前面的话:            

写到这一节的时候,CLR设计类型就已经结束了,因为CLR要求的是有一定基础的人看的,所以我们不是从基础类型以及运算符开始的,文章从一开始就讲的是深入面向对象编程,研究C#程序的设计模式。C#面向对象编程有三个特点:封装,继承,多态。接口的实现就是实现继承

其实在开始之前说一下这两天发生的事情,前几天维护项目代码时,虽然是自己写得但是由于逻辑判断比较多,有些变量名起的也不是很有意义,在看的时候就完全忘记当初为啥要写成这样了,也是有点汗颜,所以最近就把代码整洁之道也放在了看书的目录上,今后的示例代码也会符合代码整洁之道上的一些要求去写。而不是用无意义的a,b,c去做为变量名。

接口定义:

那么接口要做的事,也是让代码更加简洁的事情,接口定义了所有类继承接口时应遵循的语法合同。接口定义了语法合同 "是什么" 部分,派生类定义了语法合同 "怎么做" 部分。接口定义了属性、方法和事件,这些都是接口的成员。接口只包含了成员的声明。成员的定义是派生类的责任。接口提供了派生类应遵循的标准结构。

说了这么多,那么我们用接口实现一个计算器的例子,通过这个例子来说明接口到底是如何定义,如何使用的。先看接口的定义:

   interface ICalculator {
//通常接口命令以 I 字母开头,ICalculator 中文意思就是计算器接口
//这个接口规定了一个计算器要有基本的加减乘除运算方法
//加法
int Add(int x,int y);
//减法
int Reduce(int x, int y);
//乘法
int Ride(int x, int y);
//除法
int Except(int x, int y);
}

接口实现:

上面的代码很简单吧,在接口中不能声明静态方法,接口中的方法不能用可见性修饰符修饰,在来看这句话:接口定义了所有类继承接口时应遵循的语法合同,也就是接口先规定好了我要实现那些方法和行为,继承我的要实现(重写)我的所有方法,并且返回值要和我相同,参数也要和我相同。否则就是违反了合同,语法就会报出错误。有了合同,类现在就像是一个建筑包工头,我只需要按照你的约定做,你让我盖十层楼,我就盖十层楼。继承接口类如下:

  public class Calculator : ICalculator
{
//:表示继承了计算机接口
//实现接口中的加法
public int Add(int x, int y) {
int sum = x + y;
return sum;
}
//实现接口中的减法
public int Reduce(int x, int y)
{
int sum = x - y;
return sum;
}
//实现接口中的乘法
public int Ride(int x, int y)
{
int sum = x * y;
return sum;
}
//实现接口中的除法
public int Except(int x, int y)
{
int sum = x / y;
return sum;
}
}

接口调用: 

继承类必须实现接口中的所有方法,并且不能静态类继承接口,这里实现了接口中得所有方法,继续用上一个例子,接口好比是一个承包方:会告诉你我要什么样的房子,房子的颜色和大小,继承类就是施工方,按照承包方的要求去盖房子,知道了房子大小就开始计算要买多少砖瓦水泥。那么实现了接口的类如何调用实现呢?和正常的类一样调用和实现:

     static void Main(string[] args)
{
//实例化一个计算器
Calculator calculator1 = new Calculator();
//输出计算器的实现 输出结果为3
Console.WriteLine(calculator1.Reduce(, ));
}

这就是一个最简单的接口调用和实现。看了这么多,你可能觉得使用接口不是更加麻烦了么?如果从基类中继承不是更加简单?这样还需要定义接口,并且接口中所有成员都必须实现其方法。表明上来看是这样的,那么所有东西都有其使用的场景,接口的出生也不是为了简单类而出生。C#和CLR所支持的泛型接口为开发人员提供了许多非常出色的功能,接下来我们讨论一下泛型接口提供的一些好处

泛型接口

首先泛型接口提供了出色的编译时类型安全性,有的接口在定义的方法使用了object参数或object返回类型,在代码调用这些接口方法时,可传递任何类型的实例应用,但这通常不是我们期望的,并且使用非泛型还会造成装箱拆箱操作,今天列举C#中另外一个泛型接口IComparable,

   public interface IComparable<in T>
{
//
// 摘要:
// 比较当前对象和同一类型的另一对象。
//
// 参数:
// other:
// 与此对象进行比较的对象。
//
// 返回结果:
// 一个值,指示要比较的对象的相对顺序。返回值的含义如下:值含义小于零此对象小于 other 参数。零此对象等于 other。大于零此对象大于 other。
int CompareTo(T other);
}

这是进行大小比较的接口,接受的参数是泛型,并且泛型参数支持协变和逆变,更大程度的支持参数的灵活性,他展示了泛型接口的另外一个好处,类可以实现一个接口若干次,只要每次使用不同的类型参数,示例:

   public sealed class Comparable:IComparable<int>,IComparable<string>
{
private int m_val=;
//方法实现了IComparable<string>的CompareTo方法
public int CompareTo(string str)
{
return m_val.CompareTo(Convert.ToInt32(str));
}
//方法实现了IComparable<string>的CompareTo方法
public int CompareTo(int number)
{
return m_val.CompareTo(number);
}
}

需要说明的是,上面的代码最终在return m_val.CompareTo(Convert.ToInt32(str));这里实际调用的元数据int的CompareTo方法,因此虽然可以接收不同类型参数,但在比较的时候还是要对泛型参数进行int转化,才可以实现int的CompareTo方法。

泛型接口约束

泛型接口还可以对自身进行约束,也称作接口约束,接口约束可以将泛型类型参数约束为多个接口,这样一来,传递的参数类型必须全部接口约束。初读这段话是不是很晕?什么是将泛型类型参数约束为多个接口?为什么约束过后,传递的参数类型必须实现全部约束?全部约束指的是哪一些约束?

    public static class SomeType {
private static void Test() {
int x = ;
Guid g = new Guid();
//可以通过 因为int继承了IComparable, IFormattable, IConvertible, IComparable<Int32>, IEquatable<Int32>
M(x);
//编译不能通过,因为GUID没有继承IConvertible
//public struct Guid : IFormattable, IComparable, IComparable<Guid>, IEquatable<Guid>
M(g);
}
//M的类型参数T被约束为只支持同时实现了
//IComparable和IConvertible接口的类型
private static int M<T>(T t) where T : IComparable, IConvertible {
return ;
}
}

上面的方法,就是将参数约束了指定的接口,并且只有实现被约束的接口才能通过编译,否则就会失败。这就是接口的泛型约束。

总结:类还是接口?

首先类只能继承一个实现,而接口可以继承多个,如果多种对象类型都能做某事,就为它创建接口。

如果是简单的类型,那么就使用基类,基类实现了大量的功能,并且不用实现所有成员

CLR设计类型之接口的更多相关文章

  1. CLR - 设计类型

    前言 好记性不如烂“笔头”系列... 目录 类型基础 基元类型.引用类型和值类型 类型与成员 常量与字段 方法 类型基础 “运行时”要求每个类型最终都从System.Object 类型派生. 由于所有 ...

  2. [CLR via C#]13. 接口

    一.类和接口继承 在Microsoft.Net Framwork中,有一个名为System.Object的类,它定义了4个公共实例方法:ToString, Equals, GetHashCode和Ge ...

  3. 重温CLR(九) 接口

    对于多继承(multiple inheritance)的概念,许多程序员并不陌生,他是指一个类从两个或多个基类派生的能力.例如,假定TransmitData类的作用是发送数据,ReceiveData类 ...

  4. 【转】App架构设计经验谈:接口的设计

    App架构设计经验谈:接口的设计 App与服务器的通信接口如何设计得好,需要考虑的地方挺多的,在此根据我的一些经验做一些总结分享,旨在抛砖引玉. 安全机制的设计 现在,大部分App的接口都采用REST ...

  5. 微信小程序的Web API接口设计及常见接口实现

    微信小程序给我们提供了一个很好的开发平台,可以用于展现各种数据和实现丰富的功能,通过小程序的请求Web API 平台获取JSON数据后,可以在小程序界面上进行数据的动态展示.在数据的关键 一环中,我们 ...

  6. 编写高质量代码改善C#程序的157个建议——建议97:优先考虑将基类型或接口作为参数传递

    建议97:优先考虑将基类型或接口作为参数传递 除了公开及类型或接口外,方法的参数也应该考虑基类型或接口. 以Enumerable类型为例,它的成员方法中只要涉及需要操作集合对象的地方,都要使用IEnu ...

  7. 编写高质量代码改善C#程序的157个建议——建议96:成员应优先考虑公开基类型或接口

    建议96:成员应优先考虑公开基类型或接口 类型成员如果优先考虑公开及类型或接口,那么会让类型支持更多的应用场合. FCL中最典型的例子是集合的功能操作.集合根据功能划分有多种类型,比如List< ...

  8. Go_笔试题记录-指针与值类型实现接口的区别

    1.如果Add函数的调用代码为: func main() { var a Integer = 1 var b Integer = 2 var i interface{} = &a sum := ...

  9. 使用suds模块进行封装,处理webservice类型的接口

    import json from suds.client import Client class HandleWebservice: ''' 定义一个webservice类型的接口处理类 ''' de ...

随机推荐

  1. ABP增删改查代码片段

    @using System.Web.Optimization @using MultiPageSimpleTask.Entitys.Dtos; @model IList<ProductDto&g ...

  2. Eclipse自动补全增强

    在Eclipse中,从Window -> preferences -> Java -> Editor -> Content assist -> Auto-Activati ...

  3. Ubuntu 中登录相关的日志

    登录相关的日志涉及到系统的安全,所以是系统管理中非常重要的一部分内容.本文试图对登录相关的日志做一个整理. /var/log/auth.log 这是一个文本文件,记录了所有和用户认证相关的日志.无论是 ...

  4. 如何删除错误提交的 git 大文件

    早上小伙伴告诉我,他无法拉下代码,我没有在意.在我开始写代码的时候,发现我的 C 盘炸了.因为我的磁盘是苏菲只有 256G 放了代码就没空间了,于是我查找到了原来是我的代码占用了居然有 2000+M ...

  5. Python学习笔记(八)

    Python学习笔记(八): 复习回顾 递归函数 内置函数 1. 复习回顾 1. 深浅拷贝 2. 集合 应用: 去重 关系操作:交集,并集,差集,对称差集 操作: 定义 s1 = set('alvin ...

  6. TQ2440--nandflash(K9F2G08U0A)驱动编写

    一.数据手册相关内容 1.地址传输周期 2.命令表 3.在寄存器中,会涉及TACLS,TWRPH0,TWRPH1的设定 这里我们就去看nandflash的数据手册 在这里我们可以清楚的看到,TACLS ...

  7. Fastify 系列教程二 (中间件、钩子函数和装饰器)

    Fastify 系列教程: Fastify 系列教程一 (路由和日志) Fastify 系列教程二 (中间件.钩子函数和装饰器) 中间件 Fastify 提供了与 Express 和 Restify ...

  8. PHP程序员的技术成长之路规划

    按照了解的很多PHP/LNMP程序员的发展轨迹,结合个人经验体会,抽象出很多程序员对未来的迷漫,特别对技术学习的盲目和慌乱,简单梳理了这个每个阶段PHP程序员的技术要求,来帮助很多PHP程序做对照设定 ...

  9. Linux系统查找

    1. which:在当前用户环境变量path指定的路径下查找可执行程序/文件. 特点:(1)只在当前用户环境变量指定的路径下查找: (2)只找出可执行程序/文件的位置: (3)查找速度非常快. 注:使 ...

  10. hibernate的操作Blob和Clob类型数据(笔记)