C#隐式/显示实现接口方法详解

作者:我是攻城狮 时间:2022-01-02 02:06:30 

接口定义了一系列的行为规范,为类型定义一种Can-Do的功能。例如,实现IEnumerable接口定义了GetEnumerator方法,用于获取一个枚举数,该枚举数支持在集合上进行迭代,也就是我们常说的foreach。接口只是定义行为,具体的实现需要由具体类型负责,实现接口的方法又分为隐式实现与显示实现。

一、隐式/显示实现接口方法

简单的说,我们平时“默认”使用的都是隐式的实现方式。例如:


interface ILog
{
 void Log();
}

public class FileLogger : ILog
{
 public void Log()
 {
   Console.WriteLine("记录到文件!");
 }
}

隐式实现很简单,通常我们约定接口命名以 I 开头,方便阅读。接口内的方法不需要用public,编译器会自动加上。类型中实现接口的方法只能是public,也可以定义成虚方法,由子类重写。现在看显示实现的方式:


public class EventLogger : ILog
{
 void ILog.Log()
 {
   Console.WriteLine("记录到系统事件!");
 }
}

与上面不同的是,方法用了ILog指明,而且没有(也不能有)public或者private修饰符。

除了语法上的不同,调用方式也不同,显示实现只能用接口类型的变量来调用,如:


FileLogger fileLogger = new FileLogger();
fileLogger.Log(); //正确
EventLogger eventLogger = new EventLogger();      
eventLogger.Log(); //报错
ILog log = new EventLogger();
log.Log(); //正确

二、何时使用

1. c#允许实现多个接口,如果多个接口定义了相同的方法,可以用显示实现的方式加以区分,例如:


interface ISendable
{
 void Log();
}

public class EmailLogger : ILog, ISendable
{
 void ILog.Log()
 {
   Console.WriteLine("ILog");
 }

void ISendable.Log()
 {
   Console.WriteLine("ISendable");
 }
}

2. 增强编译时的类型安全和避免值类型装箱

有了泛型,我们自然可以做到编译时的类型安全和避免值类型装箱的操作。但有时候可能没有对应的泛型版本。例如:IComparable(这里只是举例,实际有IComparable<T>)。如:


interface IComparable
{
 int CompareTo(object obj);
}

struct ValueType : IComparable
{
 private int x;
 public ValueType(int x)
 {
   this.x = x;
 }

public int CompareTo(object obj)
 {
   return this.x - ((ValueType)obj).x;
 }
}

调用:


ValueType vt1 = new ValueType(1);
ValueType vt2 = new ValueType(2);
Console.WriteLine(vt1.CompareTo(vt2));

由于形参是object,上面的CompareTo会发生装箱;而且无法获得编译时的类型安全,例如我们可以随便传一个string,编译不会报错,等到运行时才抛出InvalidCastException。使用显示实现接口的方式,如:


public int CompareTo(ValueType vt)
{
 return this.x - vt.x;
}

int IComparable.CompareTo(object obj)
{
 return CompareTo((ValueType)obj);
}

再次执行上面的代码,就不会发生装箱操作,而且可以获得编译时的类型安全了。但是如果我们用接口变量调用,就会再次发生装箱并丧失编译时的类型安全检测能力。


IComparable vt1 = new ValueType(1); //装箱
ValueType vt2 = new ValueType(2);
Console.WriteLine(vt1.CompareTo(vt2)); //再次装箱

三、缺点

1. 显示实现只能用接口类型变量调用,会给人的感觉是某类型实现了该接口却无法调用接口中的方法。特别是写成类库给别人调用时,显示实现的接口方法在vs中按f12都不会显示出来。(这点有人在csdn提问过,为什么某个类型可以不用实现接口方法)

2. 对于值类型,要调用显示实现的方法,会发生装箱操作。

3. 无法被子类继承使用。

标签:C#,接口
0
投稿

猜你喜欢

  • Unity3D游戏开发数据持久化PlayerPrefs的用法详解

    2022-11-11 23:16:00
  • 在C#程序中对MessageBox进行定位的方法

    2022-03-04 18:45:26
  • Java删除二叉搜索树的任意元素的方法详解

    2021-10-04 12:27:26
  • Maven 配置文件 生命周期 常用命令详解

    2022-07-05 19:59:39
  • Kotlin基础教程之Run,标签Label,函数Function-Type

    2022-08-28 14:11:01
  • Java设计模式之代理模式_动力节点Java学院整理

    2021-08-24 05:55:18
  • springboot 启动如何排除某些bean的注入

    2022-01-20 18:36:01
  • java实现的各种排序算法代码示例

    2023-01-29 03:48:27
  • 分析JAVA中几种常用的RPC框架

    2022-12-11 03:54:18
  • JUC系列学习工具类CountDownLatch详解

    2023-10-01 12:19:23
  • Java 详解包装类Integer与int有哪些共通和不同

    2022-06-15 20:12:18
  • springcloud项目占用内存好几个G导致服务器崩溃的问题

    2023-03-30 09:54:25
  • 图解Java排序算法之快速排序的三数取中法

    2022-02-26 17:58:23
  • C#实现输入10个数存入到数组中并求max和min及平均数的方法示例

    2023-11-30 05:58:54
  • java数独游戏完整版分享

    2023-05-22 07:09:56
  • spring boot + mybatis实现动态切换数据源实例代码

    2021-11-14 03:37:59
  • Java 线程池ThreadPoolExecutor源码解析

    2022-06-28 09:13:59
  • 对Java ArrayList的自动扩容机制示例讲解

    2022-04-09 07:58:25
  • 详解Spring 中如何控制2个bean中的初始化顺序

    2023-05-06 13:43:18
  • JAVA十大排序算法之桶排序详解

    2022-11-08 01:07:47
  • asp之家 软件编程 m.aspxhome.com