[C#学习笔记]接口的特性与用法
视频地址:一期视频看透C#接口的全部特性及用法_哔哩哔哩_bilibili
强烈推荐学习C#和WPF的朋友关注此UP,知识点巨多,讲解透彻!
一、总览
public interface IOverall
{/// <summary>/// 最普通的方法/// </summary>void Foo();/// <summary>/// 属性/// </summary>string Name { get;set; }/// <summary>/// 索引器/// </summary>/// <param name="index"></param>/// <returns></returns>int this[int index] { get; set; }/// <summary>/// 事件/// </summary>event EventHandler OnNameChanged;/// <summary>/// 带默认实现的方法/// </summary>void Bar() => Console.WriteLine("Bar");/// <summary>/// 私有方法(需要带默认实现)/// </summary>private void NonPublicMethod1() => Console.WriteLine("Private");/// <summary>/// 受保护方法(需要带默认实现,或者可以不实现,继承后实现)/// </summary>protected void NonPublicMethod2() => Console.WriteLine("Protected");/// <summary>/// 静态方法(需要带默认实现)/// </summary>static void StaticMethod() => Console.WriteLine("Static");/// <summary>/// 抽象静态方法/// </summary>static abstract void AbstractStaticMethod();/// <summary>/// 静态虚方法(需要带默认实现)/// </summary>static virtual void VirtualStaticMethod() => Console.WriteLine("Virtual Static");
}
二、带默认实现的方法
C# 8.0引入
如果接口的方法提供了一个默认实现,那么实现该接口的类可以不实现此方法。
如果实现了此方法,则会覆盖接口中的默认实现,可以理解为接口中带默认实现的方法其实是virtual,而类中的如果实现了此方法,其实是override。
2.1 用法
参考以下代码:
Test test = new Test();
((IFoo1)test).Foo1();
((IFoo1)test).Foo2();public interface IFoo1
{ void Foo1(){$"这是{nameof(IFoo1)}中{nameof(Foo1)}的默认实现".Dump();}void Foo2(){$"这是{nameof(IFoo1)}中{nameof(Foo2)}的默认实现".Dump();}
}class Test : IFoo1
{ public void Foo1(){$"这是{nameof(Test)}中{nameof(Foo1)}的实际实现".Dump();}
}
输出结果为
这是Test中Foo1的实际实现
这是IFoo1中Foo2的默认实现
对于带默认实现的方法,如果类中不实现此方法,调用时需要先将类转换成接口,再调接口上的方法,参考以下代码:
Test test = new Test();
((IFoo1)test).Foo1();
((IFoo2)test).Foo1();public interface IFoo1
{void Foo1(){$"这是{nameof(IFoo1)}中{nameof(Foo1)}的默认实现".Dump();}
}public interface IFoo2
{void Foo1(){$"这是{nameof(IFoo2)}中{nameof(Foo1)}的默认实现".Dump();}
}class Test : IFoo1,IFoo2
{ }
接口IFoo1和IFoo2都有一个带默认实现的方法,且名称相同,类Test实现了IFoo1和IFoo2,如果直接能访问到接口中的方法Foo1,就会造成冲突。
2.2 应用场景
在不破坏影响已有实现的情况下,可以添加新成员。这解决了在第三方已经大量使用了的接口上进行扩展带来问题的痛点。
2.3 private,带默认实现的方法
只能在接口内调用,子接口,实现接口的类中不可访问,参考以下代码
Test test = new Test();
((IFoo1)test).Foo1();public interface IFoo1
{ void Foo1(){Foo2();}private void Foo2(){$"这是{nameof(IFoo1)}中{nameof(Foo2)}的默认实现".Dump();}
}class Test : IFoo1
{ }
2.4 protected,带默认实现的方法
可以在子接口中调用或者覆盖,实现接口的类不可访问
参考以下代码:
Test1 test1 = new Test1();
((IFoo2)test1).Foo2();
((IFoo3)test1).Foo2();public interface IFoo1
{ protected void Foo1(){$"这是{nameof(IFoo1)}中{nameof(Foo1)}的默认实现".Dump();}
}public interface IFoo2 : IFoo1
{void Foo2(){Foo1();}
}public interface IFoo3 : IFoo1
{void Foo2(){Foo1();}new void Foo1(){$"这是{nameof(IFoo3)}中{nameof(Foo1)}的默认实现".Dump();}
}class Test1:IFoo2,IFoo3
{
}
运行结果
这是IFoo1中Foo1的默认实现
这是IFoo3中Foo1的默认实现
2.5 Static,带默认实现的方法
静态方法一般用于处理泛型类
参考以下代码:
var student = IDeserializable<Student>.Deserialize("{\"Id\":42,\"Name\":\"Jack\"}");
student.Dump();interface IDeserializable<T>
{static T? Deserialize(string json) => JsonSerializer.Deserialize<T>(json);
}class Student : IDeserializable<Student>
{public int Id {get;set;}public string Name { get; set; }
}
运行结果:
三、static abstract,抽象静态方法
在接口可不实现,实现此接口的类必须要实现此方法
应用场景:
3.1 约束子类具有方法Deserializable,实现序列化功能
interface IDeserializable<T>
{static abstract T Deserializable(string json);
}class MyDataModel : IDeserializable<MyDataModel>
{public static MyDataModel Deserializable(string json){return JsonSerializer.Deserialize<MyDataModel>(json);}
}
3.2 如工厂类中具有方法Create,提供创建功能
interface IFactory<T>{static abstract T Create();}class ClassToBeCreated{}class ClassWithFactoryMethod : IFactory<ClassToBeCreated>{public static ClassToBeCreated Create(){return new ClassToBeCreated();}}
3.3 约束子类实现静态单例
interface ISingleton<T> where T : ISingleton<T>{static abstract T Instance { get; }}class SingletonClass : ISingleton<SingletonClass>{private static readonly Lazy<SingletonClass> instanceHolder = new Lazy<SingletonClass>();public static SingletonClass Instance => instanceHolder.Value;}
3.4 逻辑运算法,提供自定义逻辑运算
interface IOperators<T> where T : IOperators<T>{static abstract T operator +(T left, T right);static abstract T operator -(T left, T right);}class MyNumber : IOperators<MyNumber>{public int Value { get; }public MyNumber(int value){Value = value;}public static MyNumber operator +(MyNumber left, MyNumber right){return new MyNumber(left.Value + right.Value);}public static MyNumber operator -(MyNumber left, MyNumber right){return new MyNumber(left.Value - right.Value);}}
四、static virtual 静态虚方法
通常用于泛型调用的场合
特性:
1.子类如果未实现具体方法,调用时调用接口中的方法
2.子类如果实现具体方法,调用时调用子类中的方法
代码:
TestCallerInstance(new A());
TestCallerInstance(new B());
ITestInterfaceGenric<A>.TestCallerGeneric();
ITestInterfaceGenric<B>.TestCallerGeneric();static void TestCallerInstance<T>(T t) where T : ITestInterface
{Console.WriteLine(T.TestString1());
}public interface ITestInterface
{static virtual string TestString1(){return "TestString1 ITestInterface";}
}public interface ITestInterfaceGenric<TSelf> where TSelf : ITestInterfaceGenric<TSelf>
{public static void TestCallerGeneric(){Console.WriteLine(TSelf.TestString2());}static virtual string TestString2(){return "TestString2 ITestInterfaceGeneric";}
}public class A : ITestInterface, ITestInterfaceGenric<A>
{public static string TestString2(){return "TestString A";}
}public class B : ITestInterface, ITestInterfaceGenric<B>
{public static string TestString1(){return "TestString1 ITestInterface B";}
}
运行结果:
TestString1 ITestInterface
TestString1 ITestInterface B
TestString A
TestString2 ITestInterfaceGeneric
相关文章:

[C#学习笔记]接口的特性与用法
视频地址:一期视频看透C#接口的全部特性及用法_哔哩哔哩_bilibili 强烈推荐学习C#和WPF的朋友关注此UP,知识点巨多,讲解透彻! 一、总览 public interface IOverall {/// <summary>/// 最普通的方法/// </summary>v…...
java发送邮件报错,Could not connect to SMTP host: smtp.exmail.qq.com, port: 465
发现问题 我使用的 docker 运行的 jdk 环境,服务调用发送邮件报错 javax.mail.MessagingException: Could not connect to SMTP host: smtp.exmail.qq.com, port: 465;nested exception is:javax.net.ssl.SSLHandshakeException: No appropriate protocol (protoc…...

开放式耳机有哪些好处?性价比排行前十的四款蓝牙耳机推荐
开放式耳机有以下好处: 佩戴舒适:开放式耳机不入耳,不堵塞耳道,长时间佩戴耳朵不易感到闷热和疼痛,相比传统入耳式耳机,能减少对耳道的压迫感和摩擦,让耳朵更舒适。 更健康卫生:不入…...

FreeRTOS(速记版)
第一章 初识 FreeRTOS 1.1 FreeRTOS简介 FreeRTOS 采用了 MIT 开源许可,这允许将 FreeRTOS 操作系统用于商业应用,并且不需要公开源代码。此外,FreeRTOS 还衍生出了另外两个操作系统:OpenRTOS 和 SafeRTOS,其中 OpenR…...
解锁中东市场新蓝海:Bigo社交媒体如何赋能APP广告营销优势
解锁中东市场新蓝海:Bigo社交媒体如何赋能APP广告营销优势 在全球数字化浪潮的推动下,中东地区以其独特的文化背景、高速的经济增长以及庞大的年轻消费群体,成为了众多品牌与APP开发者竞相争夺的市场高地。作为该地区颇具影响力的社交媒体平…...
【网络】DNS
definition DNS(Domain Name System,域名系统)服务器是互联网上的重要基础设施之一,它的主要作用是将人们易于记忆的域名(如www.example.com)转换成计算机可以直接识别的IP地址(如192.0.2.1&am…...

如何使用ChatGPT,完成学术论文文献综述的编写?
学境思源,一键生成论文初稿: AcademicIdeas - 学境思源AI论文写作 在学术研究中,文献综述是了解研究现状、辨识研究空白并为自己的研究奠定理论基础的关键环节。ChatGPT 可以在文献综述的编写过程中提供有效的支持,从文献搜集、批…...

探索GPU算力在大模型和高性能计算中的无限潜能
在当今科技领域,大模型和高性能计算正以惊人的速度发展。大模型如语言模型、图像识别模型等,规模越来越大,精度越来越高,能够处理复杂的任务和生成逼真的结果。高性能计算则凭借强大的计算能力,推动着科学研究、工程设…...

【信创】统信UOS图形界面登录闪退的解决方法
原文链接:【信创】统信UOS图形界面登录闪退的解决方法 Hello,大家好啊!今天给大家带来一篇关于统信UOS 1070桌面操作系统中,图形界面登录时出现闪退或输入正确的用户名和密码后又跳转回登录界面问题的解决方法的文章。这种问题可能…...

排序(插入,希尔,选择,堆,冒泡,快速,归并,计数)
本文中的Swap()函数都是下面这段代码 // 交换 void Swap(int* p1, int* p2) {int tmp *p1;*p1 *p2;*p2 tmp; }文章目录 常见排序:一.插入排序1.直接插入排序:2.希尔排序: 二.选择排序1.选择排序:2.堆排序: 三.交换排…...

【recast-navigation/源码解析】findStraightPath详解以及寻路结果贴边优化
说在前面 recast-navigation版本:1.6.0 叉积cross product 正常来讲,叉乘为: ∣ A ⃗ B ⃗ ∣ ∣ x A y A x B y B ∣ x A ⋅ y B − x B ⋅ y A |\vec{A} \times \vec{B}|\begin{vmatrix} x_A & y_A \\ x_B & y_B \end{vmatrix…...

移动管家手机智能控制汽车系统
手机可以通过下载特定的应用程序来控制汽车系统,实现远程启动、锁/解锁车门、调节车内温度等功能。 手机智能控制汽车系统主要通过下载并安装特定的APP来实现。 首先,用户需要确定自己的手机系统是安卓还是苹果版,然后前往应用…...

828华为云征文|华为云Flexus X实例Redis性能加速评测及对比
目录 前言 一、华为云Flexus X加速Redis购买 1.1 Flexus X实例购买 1.2 Redis加速镜像选择 1.3 重置密码 1.4 登录Flexus X实例 1.5 Flexus X实例Redis验证 二、Redis测评工具介绍 三、华为云Flexus X实例加速Redis测评 3.1 string类型 3.2 hash类型 3.3 list类型 3.4 set类型 …...

【OpenCV3】图像的翻转、图像的旋转、仿射变换之图像平移、仿射变换之获取变换矩阵、透视变换
1 图像的放大与缩小 2 图像的翻转 3 图像的旋转 4 仿射变换之图像平移 5 仿射变换之获取变换矩阵 6 透视变换 1 图像的放大与缩小 resize(src, dsize[, dst[, fx[, fy[, interpolation]]]]) src: 要缩放的图片dsize: 缩放之后的图片大小, 元组和列表表示均可.dst: 可选参数, 缩…...
不要认为996是开玩笑
996 预防针 随着秋招进程的不断推进,有部分同学已经 OC,有部分同学还在苦苦挣扎,并不断降低自己的预期,包括在和 HR 沟通过程中,主动说出自己愿意接受加班,愿意接受 996,以此来博得企业方面的加…...

精益工程师资格证书:2024年CLMP报名指南
随着全球对精益管理的需求日益增长,精益管理专业人士资格认证(CLMP)正成为越来越多精益工程师和精益管理人员提升职业竞争力的首选。作为一种注重管理而非生产的认证,CLMP不仅适用于制造业的专业人士,也吸引了各行业的…...

【Unity基础】如何选择脚本编译方式Mono和IL2CPP?
Edit -> Project Settings -> Player 在 Unity 中,Scripting Backend 决定了项目的脚本编译方式,即如何将 C# 代码转换为可执行代码。Unity 提供了两种主要的 Scripting Backend 选项:Mono 和 IL2CPP。它们之间的区别影响了项目的性能、…...
写在OceanBase开源三周年
我收获的深刻感触get 感触1:解决问题才有生存价值 [产品力] 感触2:永无止境的“易用性” [易用性] 感触3:立下“双赢”的flag 感触4:社区建设离不开用户和开发者参与 感触5:从易用到用户自助 [自助能力] 当时想法很简…...

【笔记】408刷题笔记
文章目录 三对角三叉树求最小带权路径UDP报文首部和TCP报文首部IP报文首部TCP报文首部UDP报文首部 刷新和再生的区别地址译码 为了区分队空队满,可以使用三种处理方式 1)牺牲一个单元 队头指针在队尾指针的下一位置作为队满的标志 队满条件:(…...

GitHub Star 数量前 13 的自托管项目清单
一个多月前,我们撰写并发布了这篇文章《终极自托管解决方案指南》。在那篇文章里我们深入探讨了云端服务与自托管方案的对比、自托管的潜在挑战、如何选择适合自托管解决方案,并深入介绍了五款涵盖不同场景的优秀自托管产品。 关于自托管的优势…...

UE5 学习系列(二)用户操作界面及介绍
这篇博客是 UE5 学习系列博客的第二篇,在第一篇的基础上展开这篇内容。博客参考的 B 站视频资料和第一篇的链接如下: 【Note】:如果你已经完成安装等操作,可以只执行第一篇博客中 2. 新建一个空白游戏项目 章节操作,重…...
Vim 调用外部命令学习笔记
Vim 外部命令集成完全指南 文章目录 Vim 外部命令集成完全指南核心概念理解命令语法解析语法对比 常用外部命令详解文本排序与去重文本筛选与搜索高级 grep 搜索技巧文本替换与编辑字符处理高级文本处理编程语言处理其他实用命令 范围操作示例指定行范围处理复合命令示例 实用技…...
日语学习-日语知识点小记-构建基础-JLPT-N4阶段(33):にする
日语学习-日语知识点小记-构建基础-JLPT-N4阶段(33):にする 1、前言(1)情况说明(2)工程师的信仰2、知识点(1) にする1,接续:名词+にする2,接续:疑问词+にする3,(A)は(B)にする。(2)復習:(1)复习句子(2)ために & ように(3)そう(4)にする3、…...
DockerHub与私有镜像仓库在容器化中的应用与管理
哈喽,大家好,我是左手python! Docker Hub的应用与管理 Docker Hub的基本概念与使用方法 Docker Hub是Docker官方提供的一个公共镜像仓库,用户可以在其中找到各种操作系统、软件和应用的镜像。开发者可以通过Docker Hub轻松获取所…...

Opencv中的addweighted函数
一.addweighted函数作用 addweighted()是OpenCV库中用于图像处理的函数,主要功能是将两个输入图像(尺寸和类型相同)按照指定的权重进行加权叠加(图像融合),并添加一个标量值&#x…...

《用户共鸣指数(E)驱动品牌大模型种草:如何抢占大模型搜索结果情感高地》
在注意力分散、内容高度同质化的时代,情感连接已成为品牌破圈的关键通道。我们在服务大量品牌客户的过程中发现,消费者对内容的“有感”程度,正日益成为影响品牌传播效率与转化率的核心变量。在生成式AI驱动的内容生成与推荐环境中࿰…...
Python如何给视频添加音频和字幕
在Python中,给视频添加音频和字幕可以使用电影文件处理库MoviePy和字幕处理库Subtitles。下面将详细介绍如何使用这些库来实现视频的音频和字幕添加,包括必要的代码示例和详细解释。 环境准备 在开始之前,需要安装以下Python库:…...

Android 之 kotlin 语言学习笔记三(Kotlin-Java 互操作)
参考官方文档:https://developer.android.google.cn/kotlin/interop?hlzh-cn 一、Java(供 Kotlin 使用) 1、不得使用硬关键字 不要使用 Kotlin 的任何硬关键字作为方法的名称 或字段。允许使用 Kotlin 的软关键字、修饰符关键字和特殊标识…...

保姆级教程:在无网络无显卡的Windows电脑的vscode本地部署deepseek
文章目录 1 前言2 部署流程2.1 准备工作2.2 Ollama2.2.1 使用有网络的电脑下载Ollama2.2.2 安装Ollama(有网络的电脑)2.2.3 安装Ollama(无网络的电脑)2.2.4 安装验证2.2.5 修改大模型安装位置2.2.6 下载Deepseek模型 2.3 将deepse…...

20个超级好用的 CSS 动画库
分享 20 个最佳 CSS 动画库。 它们中的大多数将生成纯 CSS 代码,而不需要任何外部库。 1.Animate.css 一个开箱即用型的跨浏览器动画库,可供你在项目中使用。 2.Magic Animations CSS3 一组简单的动画,可以包含在你的网页或应用项目中。 3.An…...