C# 8: 默认接口方法

翻译自 John Demetriou 2018年8月4日 的文章 《C# 8: Default Interface Methods》[1]

C# 8 之前

今天我们来聊一聊默认接口方法。听起来真的很奇怪,不是吗?接口仅用于定义契约。接口的实现类会拥有一组公共方法,不过实现类被赋予了以其自己的方式实现每个方法的自由。目前为止,如果我们还需要为这些方法中的一个或多个方法提供实现,我们将使用继承。
如果我们希望这个类不是实现所有方法,而只是实现其中的一个子集,我们可以将这些方法和类本身抽象(abstract)。

例如,我们不能这么写:

interface IExample
{void Ex1();                                      // 允许void Ex2() => Console.WriteLine("IExample.Ex2"); // 不允许(C# 8 以前)
}

我们不得不用下面的抽象类来替代:

abstract class ExampleBase
{public abstract void Ex1();public void Ex2() => Console.WriteLine("ExampleBase.Ex2");
}

不过还好,这已经足够满足我们的大部分需求了。

C# 8 之后

那么,有什么改变吗?为什么我们需要引入这个新特性?我们错过了什么并且从未注意到我们错过了什么?

菱形问题

由于菱形问题[2],C#(以及许多其他语言)不支持多重继承。为了允许多重继承,同时避免菱形问题,C# 8 引入了默认接口方法。

从 C# 8 开始,使用默认接口方法,您可以拥有一个接口定义,以及该定义中某些或所有方法的默认实现。

interface IExample
{void Ex1();                                      // 允许void Ex2() => Console.WriteLine("IExample.Ex2"); // 允许
}

因此,现在您可以实现一个含有已实现方法的接口,并且可以避免希望从特定类(也包含通用方法)继承的类中的代码重复。

使用默认接口方法,菱形问题并没得到百分之百解决。当一个类继承自从第三个接口继承而来的两个接口,并且所有接口都实现了相同方法时,仍然可能发生这种情况。
在这种情况下,C# 编译器将根据当前上下文选择调用适当的方法。如果无法推断出特定的哪一个,则会显示编译错误。

例如,假设我们有以下接口:

interface IA
{void DoSomething();
}interface IB : IA
{void DoSomething() => Console.WriteLine("I am Interface B");
}interface IC : IA
{void DoSomething() => Console.WriteLine("I am Interface C");
}

然后,我创建一个实现上述两个接口的类 D,会引发一个编译错误:

//编译器提示:“D”未实现接口成员“IA.DoSomething()”
public class D : IB, IC
{ }

但是,如果类 D 实现它自己版本的 DoSomething 方法,那么编译器将知道调用哪个方法:

public class D : IB, IC
{public void DoSomething() => Console.WriteLine("I am Class D");
}

若 Main 方法代码如下:

static void Main()
{var x = new D();x.DoSomething();Console.ReadKey();
}

运行程序,控制台窗口输出:I am Class D

其他益处

使用方法的默认接口实现,API 提供者可以扩展现有接口而不破坏遗留代码的任何部分。

Trait 模式

译者注:
在计算机编程中,特征(Trait)是面向对象编程中使用的一个概念,它表示可用于扩展类的功能的一组方法。[3]

Trait 模式大体上就是多个类需要的一组方法。
在此之前,C# 中的 Trait 模式是使用抽象类实现的。但是由于多重继承不可用,实现 Trait 模式变得非常棘手,所以大多数人要么避开它,要么迷失在一个巨大的继承链中。

不过,在接口中使用默认方法实现,这将发生改变。我们可以通过在接口中使用默认接口方法实现,提供一组需要类拥有的方法,然后让这些类继承此接口。
当然,任何一个类都可以用它们自己的实现覆盖这些方法,但是以防它们不希望这么做,我们为它们提供了一组默认的实现。

以下为译者补充

接口中的具体方法

默认接口方法的最简单形式是在接口中声明具体方法,该方法是具有主体部分的方法。

interface IA
{void M() { Console.WriteLine("IA.M"); }
}

实现此接口的类不必实现其具体方法。

class C : IA { } // OKstatic void Main()
{IA i = new C();i.M(); // 输出 "IA.M"
}

类 C 中 IA.M 的最终替代是在 IA 中声明的具体方法 “M” 。
请注意,类只能实现接口,而不会从接口继承成员

C c = new C(); // 或者 var c = new C();
c.M();         // 错误: 类 'C' 不包含 'M' 的定义

但如果实现此接口的类也实现了具体方法,则同一般的接口含义是一样的:

class C : IA
{public void M() { Console.WriteLine("C.M"); }
}static void Main()
{IA i = new C();i.M(); // 输出 "C.M"
}

相关链接:

  1. https://www.devsanon.com/c/c-8-default-interface-methods/ C# 8: Default Interface Methods ↩︎

  2. https://mp.weixin.qq.com/s/EZ_jIjT6hYFrhbJ9BZ7Amw 菱形问题 ↩︎

  3. https://en.wikipedia.org/wiki/Trait_(computer_programming) Trait ↩︎

作者 :John Demetriou 

译者 :技术译民 
出品 :技术译站(https://ITTranslator.cn/)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/307262.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

跟我一起学.NetCore之MVC过滤器,这篇看完走路可以仰着头走

前言MVC过滤器在之前Asp.Net的时候就已经广泛使用啦,不管是面试还是工作,总有一个考点或是需求涉及到,可以毫不疑问的说,这个技术点是非常重要的; 在之前参与的面试中,得知很多小伙伴只知道有一两个过滤器&…

彭博社:中企游说政府否决英伟达收购Arm

9月14日,英伟达与软银宣布达成最终协议,英伟达将以400亿美元的价格从软银收购ARM。在台积电明确于9月15日后不再向HW的ARM提供代工服务之后,一家美国公司又将ARM收入囊中,这使国产ARM CPU在设计和制造环节都存在巨大风险&#xff…

Explain各个字段的含义

文章目录[TOC]1:expanin的结果示例2:各个字段的含义(1):id(2):select_type(3):table(4):type(重要 我们利用索引查找出来的记录显示)(5): possible_keys(6):key(7):key_len(8):ref(8):rows(重要)(9):extra1:expanin的结果示例 mysql> explain select…

asp.net core监控—引入Prometheus(三)

上一篇博文中说到Prometheus有四种指标类型:Counter(计数器)、Gauge(仪表盘)、Histogram(直方图)、Summary(摘要),并且我们做了一个Counter的Demo&#xff0c…

秋招日常(一天三封感谢信)

就离谱 上午收到两封 搞得复习贼没心态 下午打完球回来一看手机又收到一封 这个有点伤 自己测评完 笔试 笔试完 约面 技术面过了 hr又面 面完说等消息 我寻思hr面能有啥可挂的 就离谱 一想到自己经过那么多轮 还是挂了 就难受 什么鬼 求求了 给个offer吧 要不我快成了天天向下的…

asp.net core监控—引入Prometheus(一)

Prometheus是CNCF毕业的第二个项目,算是明星产品(可自行了解Prometheus的功能),asp.net core当然不能错过与之配套使用。在.net中是通过prometheus.net【https://github.com/prometheus-net/prometheus-net】引入的。上图是用Prom…

渣渣双非秋招以来第一份offer

一:10月25日投递 boss上进行投递 二:10月27一面 1:在项目当中中如何运用的线程池2:运用线程池的话可以解决什么问题3:项目当中那些部分是运用了一些异步的操作4:项目当中那些部分是运用了一些异步的操作5:java并发编程 信号量 。。。6.JDK中有无安全的ArrayList&…

1024福利 | .NET社区联合清华大学出版社送好书来啦!

话不多说,直入主题。.NET社区各大公众号联合清华大学出版社为粉丝送上1024的福利,还请诸位笑纳!!!有哪些书呢?.NET并发编程实战 作者:[美] 里卡尔多特雷尔(Riccardo Terrell&#xf…

asp.net core监控—引入Prometheus(二)

上一篇博文中,说明了怎么引进Prometheus到asp.net core项目中,因为是Demo,所以Prometheus和Grafana都是windows版本,本地执行的,生产环境上这些服务可以根据的公司的架构,放到适合的环境内,现在…

PTA 数据结构与算法题目集(中文)

一:数据结构与算法题目(中文版) 7-2 一元多项式的乘法与加法运算 (20 分) 7-3 树的同构 (25 分) 7-4 是否同一棵二叉搜索树 (25 分) 7-6 列出连通集 (25 分)(详解) 7-7 六度空间 (30 分) 7-8 哈利波特的考试 (25 分) 7-14 电话聊天狂人 (25 …

ASP.NET Core Blazor Webassembly 之 渐进式应用(PWA)

Blazor支持渐进式应用开发也就是PWA。使用PWA模式可以使得web应用有原生应用般的体验。什么是PWAPWA应用是指那些使用指定技术和标准模式来开发的web应用,这将同时赋予它们web应用和原生应用的特性。例如,web应用更加易于发现——相比于安装应用&#xf…

TCP三次握手(详解)

一:引出 客户端与服务器之间数据的发送和返回的过程当中需要创建一个叫TCP connection的东西;由于TCP不存在连接的概念,只存在请求和响应,请求和响应都是数据包,它们之间都是经过由TCP创建的一个从客户端发起&#xf…

进击吧! Blazor !第五期 安全

Blazor 是一个 Web UI 框架,可通过 WebAssembly 在任意浏览器中运行 .Net 。Blazor 旨在简化快速的单页面 .Net 浏览器应用的构建过程,它虽然使用了诸如 CSS 和 HTML 之类的 Web 技术,但它使用 C#语言和 Razor 语法代替 JavaScrip…

有一种无耻叫 “我是为你着想”

这是头哥侃码的第221篇原创十一之后,我的状态一直不好,总感觉胸口有东西堵着,时不时还会觉得呼吸困难。上周末,我咨询了几位有医学背景的朋友,有说是因为季节转换而造成的身体不适,不用在意,但也…

ASP.NET Core托管运行Quartz.NET作业调度详解

Quartz.NET这么NB的作业调度系统,不会还行?今天介绍一下Quartz.NET的托管运行,官网传送门。一、前言Quartz.NET,按官网上的说法,是一款功能齐全的任务调度系统,从小型应用到大型企业级系统都能适用。在众多…

asp.net core监控—引入Prometheus(四)

上一篇博文中说到Prometheus有四种指标类型:Counter(计数器)、Gauge(仪表盘)、Histogram(直方图)、Summary(摘要),并且我们做了一个Counter的Demo&#xff0c…

[开源] .Net ORM FreeSql 1.10.0 稳步向前

写在开头FreeSql 是 .NET 开源生态下的 ORM 轮子,转眼快两年了,说真的开源不容易(只有经历过才明白)。今天带点干货和湿货给大家,先说下湿货。认识我的人,知道 CSRedisCore 是我写的另外一个开源组件&#…

CAP-微服务间通信实践

微服务间通信常见的两种方式由于微服务架构慢慢被更多人使用后,迎面而来的问题是如何做好微服务间通信的方案。我们先分析下目前最常用的两种服务间通信方案。gRPC(rpc远程调用)gRPC-微服务间通信实践场景:A服务主动发起请求到B服…

软件构造学习笔记-第二周

本周课程把第六章测试的内容提前讲了一部分,主要为实验1服务,讲了有关测试的概念、作用和基本方法。“测试优先”的思想是非常重要的,根据spec写出简单而全面的测试,在方法/类完成后第一时间对其进行测试,保证每个方法…