C# 图解教程 第5版 —— 第18章 泛型

文章目录

    • 18.1 什么是泛型
    • 18.2 C# 中的泛型
    • 18.3 泛型类
      • 18.3.1 声明泛型类
      • 18.3.2 创建构造类型
      • 18.3.3 创建变量和实例
      • 18.3.4 使用泛型的示例
      • 18.3.5 比较泛型和非泛型栈
    • 18.4 类型参数的约束
      • 18.4.1 Where 子句
      • 18.4.2 约束类型和次序
    • 18.5 泛型方法
      • 18.5.1 声明泛型方法
      • 18.5.2 调用泛型方法
      • 18.5.3 泛型方法的示例(*)
    • 18.6 扩展方法和泛型类
    • 18.7 泛型结构
    • 18.8 泛型委托
    • 18.9 泛型接口
      • 18.9.1 使用泛型接口的示例(*)
      • 18.9.2 泛型接口的实现必须唯一
    • 18.10 协变和逆变
      • 18.10.1 协变(out)
      • 18.10.2 逆变
      • 18.10.3 协变和逆变的不同
      • 18.10.4 接口的协变和逆变
      • 18.10.5 关于可变性的更多内容

18.1 什么是泛型

​ 泛型可以将重构代码并且额外添加一个抽象层,是专门为多段代码在不同的数据类型上执行相同指令而设计的。

18.2 C# 中的泛型

​ 泛型不是类型,而是类型的模板。

image-20231210150031003
图 18.1 泛型是类型的模板

​ C# 提供了以下 5 种泛型:

  1. 结构
  2. 接口
  3. 委托
  4. 方法

​ 其中 1 ~ 4 是类型,5 是成员。

image-20231210150222598
图 18.2 泛型和用户定义类型

18.3 泛型类

​ 泛型类不是实际的类,而是类的模板,因此必须先从它们构建实际的类,然后创建类的引用和实例。

  1. 在某些类型上使用一个占位符来声明一个类。
  2. 为占位符提供真实类型(构造类型)。
  3. 创建构造类型的实例。
image-20231210150450484
图 18.3 从泛型类创建实例

18.3.1 声明泛型类

  1. 在类名之后放置一组尖括号。
  2. 在尖括号中用逗号分隔占位符字符串,用于表示需要提供的类型(类型参数)。
  3. 在泛型类声明的主体中使用类型参数来表示替代类型。
image-20231210150748033

18.3.2 创建构造类型

​ 声明泛型类后,就可以告诉编译器使用哪些真实类型来替代占位符,编译器将获取这些真实类型并创建构造类型(用来创建真实类对象的模板)。

image-20231210150946452 image-20231210151000389
图 18.4 为泛型类的所有类型参数提供类型实参,让编译器产生一个可以用来创建真实类对象的构造类
  • 泛型类声明上的类型参数用作类型的占位符。
  • 在创建构造类型时提供的真实类型是类型实参。
image-20231210151124508
图 18.5 类型参数与类型实参

18.3.3 创建变量和实例

image-20231210151238224

​ 和非泛型类一样,引用和实例可以分开创建。

image-20231210151335389
图 18.6 使用构造类型来创建引用和实例

18.3.4 使用泛型的示例

image-20231210151445515
图 18.7 从泛型类创建的两个构造类

18.3.5 比较泛型和非泛型栈

表 18.1 非泛型栈和泛型栈之间的区别
image-20231210151551792 image-20231210151645921
图 18.8 非泛型栈和泛型栈

18.4 类型参数的约束

​ 要让泛型更加有用,需要提供额外的信息让编译器直到参数可以接受哪些类型,这些额外的信息称为约束

18.4.1 Where 子句

  • 每个有约束的类型参数都有自己的 where 子句。
  • 如果形参有多个约束,则使用逗号分隔。
image-20231211151758652

​ 有关 where 子句的要点如下:

  1. 在类型参数列表的关闭尖括号后列出。
  2. 不使用分隔符。
  3. 可以随意次序列出。
  4. where 是上下文关键字,可以在其他上下文使用。
image-20231211151935353 image-20231211151956737

18.4.2 约束类型和次序

表 18.2 约束类型
image-20231211152026863
  • 最多只能有一个主约束,必须放在第一位。
  • 可以有任意个接口名称约束。
  • 如果存在构造函数约束,必须放在最后。
image-20231211152225282
图 18.9 如果类型参数有多个约束,则必须遵守的顺序

18.5 泛型方法

​ 泛型方法可以在泛型 / 非泛型类、结构和接口中声明。

image-20231211152327633
图 18.10 泛型方法可以声明在泛型类型和非泛型类型中

18.5.1 声明泛型方法

  • 泛型方法有两个参数列表。
    • 方法参数列表(圆括号内)。
    • 类型参数列表(尖括号内)。
  • 方法参数列表后放置可选的约束子句。
image-20231211152624577

18.5.2 调用泛型方法

image-20231211152655620

​ 编译器使用每个构造函数实例产生方法的不同版本。

image-20231211152734084
图 18.11 有两个实例的泛型方法

​ 编译器有时可以从方法参数推断类型参数。例如,对于如下的方法声明:

image-20231211152909431

​ 编译器可以从 myInt 参数的类型推断出 T 为 int,因此可以省略尖括号。

image-20231211153008926 image-20231211153019653

18.5.3 泛型方法的示例(*)

18.6 扩展方法和泛型类

​ 和非泛型类一样,泛型类的扩展方法必须满足如下条件:

  1. 声明为 static。
  2. 是静态类的成员。
  3. 第一个参数类型中必须有关键字 this,后面是扩展的泛型类的名字。

18.7 泛型结构

​ 泛型结构的规则和条件与泛型类一致。

18.8 泛型委托

image-20231211153322516

​ C# LINQ 特性大量使用泛型委托。

18.9 泛型接口

​ 泛型接口的声明和非泛型接口的声明类似,但是要在接口名称后的尖括号中放置类型参数。

18.9.1 使用泛型接口的示例(*)

18.9.2 泛型接口的实现必须唯一

​ 必须保证类型实参的组合不会在类型中产生两个重复的接口。

​ 例如,对于下面的泛型接口,会产生潜在的冲突:S 可能用作 int 类型,此时会有两个相同类型的接口,这将不被允许。

image-20231211153927361
  • 泛型结构的名称不会和非泛型冲突。

18.10 协变和逆变

18.10.1 协变(out)

​ 给出如下例子:

image-20231211154403483 image-20231211154417119

​ 我么知道,Dog 类型的变量可以作为 Animal 类型的引用,因为 DogAnimal 派生而来,这里发生了隐式类型转换。

image-20231211154436251
图 18.12 赋值兼容性意味着可以将派生类型的引用赋值给基类变量

​ 进行扩展,添加 Factory 泛型委托、MakeDog 方法,并且 MakeDog 方法可以匹配 Factory 委托。

image-20231211154720158 image-20231211154740443

​ Main 函数的第二行尝试将 Factory<Dog> 类型赋给 Factory<Animal>类型,这将产生报错。

​ 问题的原因在于,委托 Factory<Dog> 并没有从 Factory<Animal> 派生得到。

image-20231211155355402
图 18.13 赋值兼容性不使用,因为两个委托没有继承关系

​ 我们仅希望传递 DogFactory<Animal> 委托时,代码对 Dog 类型中的 Animal 部分进行操作,这并不会发生越界访问,是完全合理的。为了完成我们的期望,可以通过添加 out 关键字改变委托声明。

image-20231211155754380 image-20231211155838829
图 18.14 协变关系允许程度更高的派生类型处于返回及输出位置

18.10.2 逆变

​ 与协变相反,如果类型参数只用于方法中的输入参数,那么可以传入更高程度的派生类引用,因为委托的方法中只对其基类部分进行操作。

image-20231211160307185

​ 调用委托时,调用代码为方法 ActOnAnimal 传入的 Dog 类型的变量,而其期望的是 Animal 对象,因此可以进行操作。

image-20231211160540377
图 18.15 逆变允许程度更高的派生类型作为输入参数

18.10.3 协变和逆变的不同

image-20231211160808224
图 18.16 协变和逆变

18.10.4 接口的协变和逆变

​ 相同的原则也适用于接口。

18.10.5 关于可变性的更多内容

​ 前面的内容讲解了显式的协变和逆变。实际上,编译器可以自动识别某个已构建的委托是协变还是逆变,并且自动进行类型强制转换,但这通常发生在没有为对象的类型赋值的时候。

  • Main 第一行创建了 Factory<Animal> 类型的委托,并直接将方法 MakeDog 赋值给它。由于没有创建 Factory<Dog> 委托,因此编译器清楚这是协变关系,允许这种赋值,哪怕委托中没有 out 标识符。
  • 到 Main 第三行时,由于第二行已经创建了 Factory<Dog> 委托,因此后面的协变关系赋值需要 out 标识符才能完成。
image-20231211161116413 image-20231211161140827
  • 可变性只适用于引用类型,不使用与值类型。
  • in、out 关键字的显式变化只适用于委托和接口,不适用于类、结构和方法。
  • 不使用 int、out 关键字的委托和接口类型参数是不变的。
image-20231211161921422

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

青少年CTF-Misc(持续更新中)

FLAG&#xff1a;当觉得自己很菜的时候&#xff0c;就静下心来学习 专研方向:Web安全&#xff0c;CTF 每日emo&#xff1a;听一千遍反方向的钟&#xff0c;我们能回到过去吗&#xff1f; 1.StegoTXT&#xff1a; 解压缩文件。发现字母中存在覆盖。使用0宽隐写在线解密得到flag…

Slate基础使用说明

目录 Slate基础使用说明 1. 简单教程 2. 要点说明 2.1 TCommands以及TCommands基类 2.2 FUICommandInfo 2.3 FUICommandList 2.4 FUIAction 2.5 UICommand 3. 代码源码 4. 工具使用 4.1 Display Ul Extension Points 4. 参考文章 Slate基础使用说明 1.…

设计模式02创建者模式

创建者模式 参考网课:黑马程序员Java设计模式详解 博客笔记 创建型模式的主要关注点是“怎样创建对象&#xff1f;”&#xff0c;它的主要特点是“将对象的创建与使用分离”。 这样可以降低系统的耦合度&#xff0c;使用者不需要关注对象的创建细节。 创建型模式分为&#…

下一代Wi-Fi技术:Wi-Fi 7(IEEE 802.11be EHT)

文章目录 Wi-Fi 7名词解释Wi-Fi 7的产生背景Wi-Fi 7的发布时间Wi-Fi 7的技术优势Wi-Fi 7 vs Wi-Fi 6Wi-Fi 7支持的新特性支持最大320MHz带宽引入更高阶的4096-QAM调制技术MIMO 1616引入Multi-Link多链路机制Multi-RUPreamble Puncturing Wi-Fi 7的应用场景推荐阅读 Wi-Fi 7名词…

DevEco Studio 生成HPK文件

DevEco Studio 生成HPK文件 一、安装环境 操作系统: Windows 10 专业版 IDE:DevEco Studio 3.1 SDK:HarmonyOS 3.1 二、生成HPK文件 生成的HPK文件存放在entry文件夹下。下图是未生成HPK的样式。 生成HPK&#xff1a;菜单Build->Build Hap(s)/APP(s)->Build Hap(s)…

Python使用分段函数拟合数据

Python使用分段函数拟合数据 前言前提条件相关介绍实验环境使用分段函数拟合数据代码实现输出结果 前言 由于本人水平有限&#xff0c;难免出现错漏&#xff0c;敬请批评改正。更多精彩内容&#xff0c;可点击进入Python日常小操作专栏、OpenCV-Python小应用专栏、YOLO系列专栏…

HCIA-H12-811题目解析(3)

1、【单选题】 以下关于路由器的描述&#xff0c;说法错误的是&#xff1f; 2、【单选题】某网络工程师在输入命令行时提示如下信息&#xff1a;Error:Unrecognized command foun at position.对于该提示信息说法正确的是&#xff1f; 3、【单选题】如下图所示的网络&#xf…

Vue3-03-reactive() 响应式基本使用

reactive() 的简介 reactive() 是vue3 中进行响应式状态声明的另一种方式&#xff1b; 但是&#xff0c;它只能声明 【对象类型】的响应式变量&#xff0c;【不支持声明基本数据类型】。reactive() 与 ref() 一样&#xff0c;都是深度响应式的&#xff0c;即对象嵌套属性发生了…

数据科学工作的20个Pandas函数(备忘)

Pandas 是数据科学社区中使用最广泛的库之一&#xff0c;它是一个强大的工具&#xff0c;可以进行数据操作、清理和分析。 本文将提供最常用的 Pandas 函数以及如何实际使用它们的样例。我们将涵盖从基本数据操作到高级数据分析技术的所有内容&#xff0c;到本文结束时&#xf…

还在为论文焦虑?免费AI写作大师帮你三分钟搞定

先来看1分钟的视频&#xff0c;对于要写论文的你来说&#xff0c;绝对有所值&#xff01; 还在为写论文焦虑&#xff1f;免费AI写作大师来帮你三步搞定 第一步&#xff1a;输入关键信息 第二步&#xff1a;生成大纲 稍等片刻后&#xff0c;专业大纲生成&#xff08;由于举例&am…

WPS没保存关闭了怎么恢复数据?3个方法,完成数据恢复!

“我今天在使用WPS时&#xff0c;突然有点急事出去了一趟&#xff0c;但是我忘记保存文档了&#xff0c;回来之后发现电脑自动关机了&#xff0c;我的文档也没了&#xff01;这可怎么办呢&#xff1f;有什么办法可以找回这些数据吗&#xff1f;” 在快节奏的工作中&#xff0c;…

PyQt6 表单布局Form Layout (QFormLayout)

锋哥原创的PyQt6视频教程&#xff1a; 2024版 PyQt6 Python桌面开发 视频教程(无废话版) 玩命更新中~_哔哩哔哩_bilibili2024版 PyQt6 Python桌面开发 视频教程(无废话版) 玩命更新中~共计43条视频&#xff0c;包括&#xff1a;2024版 PyQt6 Python桌面开发 视频教程(无废话版…

Phong vs. BRDF

在深入探讨 BRDF 和照明模型的概念之前&#xff0c;我们将介绍一种用于模拟有光泽&#xff08;glossy&#xff09;表面&#xff08;例如塑料球&#xff09;外观的技术。 从那里开始&#xff0c;推广该技术将变得更加容易&#xff0c;这就是 BRDF 和照明或反射模型概念的全部内容…

mysql:用SHOW COLUMNS FROM显示一个表的列信息

可以使用命令SHOW COLUMNS FROM table_name;显示一个表的列信息&#xff0c;例如&#xff1a;

电工--半导体器件

目录 半导体的导电特性 PN结及其单向导电性 二极管 稳压二极管 双极型晶体管 半导体的导电特性 本征半导体&#xff1a;完全纯净的、晶格完整的半导体 载流子&#xff1a;自由电子和空穴 温度愈高&#xff0c;载流子数目愈多&#xff0c;导电性能就愈好 型半导体&…

Springboot内置Tomcat线程数优化

Springboot内置Tomcat线程数优化 # 等待队列长度&#xff0c;默认100。队列也做缓冲池用&#xff0c;但也不能无限长&#xff0c;不但消耗内存&#xff0c;而且出队入队也消耗CPU server.tomcat.accept-count1000 # 最大工作线程数&#xff0c;默认200。&#xff08;4核8g内存…

Qt 线程

&#x1f4a1; 进度条显示拷贝进度&#xff08;verson 1&#xff09; 窗口上放置一个按钮和一个进度条部件&#xff0c;点击按钮&#xff0c;进行拷贝操作 —— 打开对话框选择源文件&#xff0c;然后再打开一个对话框 选择 目标文件存放位置和名称。拷贝过程中进度条显示当前…

力扣经典面试题——合并区间

合并区间 https://leetcode.cn/problems/merge-intervals/description/?envTypestudy-plan-v2&envIdtop-interview-150 这题思维量一般但比较考察API的使用。 1、数组的自定义排序 2、数组的初始化定义 3、Arrays转int 通过重写Comparator的compare方法来自定义排序规则…

SpringBoot热部署

SpringBoot热部署 借鉴链接&#x1f517;&#xff1a;SpringBoot中的热部署 添加devtools依赖和pom插件 <!-- devtools 依赖 --> <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-devtools</artifactId&…

低代码核心能力表单引擎可以提高业务处理效率,降低成本的

在数字化时代&#xff0c;企业面临着海量的数据和复杂的业务需求&#xff0c;对于低代码表单的需求也逐渐增加&#xff0c;低代码表单可以提高企业的业务处理效率&#xff0c;还可以降低开发成本&#xff0c;缩短开发周期。 低代码表单应用场景​ 低代码的表单主要用于数据采集…