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# 中的泛型
泛型不是类型,而是类型的模板。
C# 提供了以下 5 种泛型:
- 类
- 结构
- 接口
- 委托
- 方法
其中 1 ~ 4 是类型,5 是成员。
18.3 泛型类
泛型类不是实际的类,而是类的模板,因此必须先从它们构建实际的类,然后创建类的引用和实例。
- 在某些类型上使用一个占位符来声明一个类。
- 为占位符提供真实类型(构造类型)。
- 创建构造类型的实例。
18.3.1 声明泛型类
- 在类名之后放置一组尖括号。
- 在尖括号中用逗号分隔占位符字符串,用于表示需要提供的类型(类型参数)。
- 在泛型类声明的主体中使用类型参数来表示替代类型。
18.3.2 创建构造类型
声明泛型类后,就可以告诉编译器使用哪些真实类型来替代占位符,编译器将获取这些真实类型并创建构造类型(用来创建真实类对象的模板)。
- 泛型类声明上的类型参数用作类型的占位符。
- 在创建构造类型时提供的真实类型是类型实参。
18.3.3 创建变量和实例
和非泛型类一样,引用和实例可以分开创建。
18.3.4 使用泛型的示例
18.3.5 比较泛型和非泛型栈
18.4 类型参数的约束
要让泛型更加有用,需要提供额外的信息让编译器直到参数可以接受哪些类型,这些额外的信息称为约束。
18.4.1 Where 子句
- 每个有约束的类型参数都有自己的 where 子句。
- 如果形参有多个约束,则使用逗号分隔。
有关 where 子句的要点如下:
- 在类型参数列表的关闭尖括号后列出。
- 不使用分隔符。
- 可以随意次序列出。
- where 是上下文关键字,可以在其他上下文使用。
18.4.2 约束类型和次序
- 最多只能有一个主约束,必须放在第一位。
- 可以有任意个接口名称约束。
- 如果存在构造函数约束,必须放在最后。
18.5 泛型方法
泛型方法可以在泛型 / 非泛型类、结构和接口中声明。
18.5.1 声明泛型方法
- 泛型方法有两个参数列表。
- 方法参数列表(圆括号内)。
- 类型参数列表(尖括号内)。
- 方法参数列表后放置可选的约束子句。
18.5.2 调用泛型方法
编译器使用每个构造函数实例产生方法的不同版本。
编译器有时可以从方法参数推断类型参数。例如,对于如下的方法声明:
编译器可以从 myInt 参数的类型推断出 T 为 int,因此可以省略尖括号。
18.5.3 泛型方法的示例(*)
18.6 扩展方法和泛型类
和非泛型类一样,泛型类的扩展方法必须满足如下条件:
- 声明为 static。
- 是静态类的成员。
- 第一个参数类型中必须有关键字 this,后面是扩展的泛型类的名字。
18.7 泛型结构
泛型结构的规则和条件与泛型类一致。
18.8 泛型委托
C# LINQ 特性大量使用泛型委托。
18.9 泛型接口
泛型接口的声明和非泛型接口的声明类似,但是要在接口名称后的尖括号中放置类型参数。
18.9.1 使用泛型接口的示例(*)
18.9.2 泛型接口的实现必须唯一
必须保证类型实参的组合不会在类型中产生两个重复的接口。
例如,对于下面的泛型接口,会产生潜在的冲突:S 可能用作 int 类型,此时会有两个相同类型的接口,这将不被允许。
- 泛型结构的名称不会和非泛型冲突。
18.10 协变和逆变
18.10.1 协变(out)
给出如下例子:
我么知道,Dog 类型的变量可以作为 Animal 类型的引用,因为 Dog 由 Animal 派生而来,这里发生了隐式类型转换。
进行扩展,添加 Factory 泛型委托、MakeDog 方法,并且 MakeDog 方法可以匹配 Factory 委托。
Main 函数的第二行尝试将 Factory<Dog> 类型赋给 Factory<Animal>类型,这将产生报错。
问题的原因在于,委托 Factory<Dog> 并没有从 Factory<Animal> 派生得到。
我们仅希望传递 Dog 给 Factory<Animal> 委托时,代码对 Dog 类型中的 Animal 部分进行操作,这并不会发生越界访问,是完全合理的。为了完成我们的期望,可以通过添加 out 关键字改变委托声明。
18.10.2 逆变
与协变相反,如果类型参数只用于方法中的输入参数,那么可以传入更高程度的派生类引用,因为委托的方法中只对其基类部分进行操作。
调用委托时,调用代码为方法 ActOnAnimal 传入的 Dog 类型的变量,而其期望的是 Animal 对象,因此可以进行操作。
18.10.3 协变和逆变的不同
18.10.4 接口的协变和逆变
相同的原则也适用于接口。
18.10.5 关于可变性的更多内容
前面的内容讲解了显式的协变和逆变。实际上,编译器可以自动识别某个已构建的委托是协变还是逆变,并且自动进行类型强制转换,但这通常发生在没有为对象的类型赋值的时候。
- Main 第一行创建了
Factory<Animal>类型的委托,并直接将方法 MakeDog 赋值给它。由于没有创建Factory<Dog>委托,因此编译器清楚这是协变关系,允许这种赋值,哪怕委托中没有 out 标识符。 - 到 Main 第三行时,由于第二行已经创建了
Factory<Dog>委托,因此后面的协变关系赋值需要 out 标识符才能完成。
- 可变性只适用于引用类型,不使用与值类型。
- in、out 关键字的显式变化只适用于委托和接口,不适用于类、结构和方法。
- 不使用 int、out 关键字的委托和接口类型参数是不变的。

相关文章:
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 调用…...
保障事务隔离级别的关键措施
目录 引言 1. 锁机制的应用 2. 多版本并发控制(MVCC)的实现 3. 事务日志的记录与恢复 4. 数据库引擎的实现策略 结论 引言 事务隔离级别是数据库管理系统(DBMS)中的一个关键概念,用于控制并发事务之间的可见性。…...
Docker导入导出镜像、导入导出容器的命令详解以及使用的场景
一、Docker 提供用于管理镜像和容器命令 1.1 docker save 与 docker load 这是一对操作,用于处理 Docker 镜像。这个操作会将所有的镜像层以及元数据打包到一个 tar 文件中。然后,你可以使用 docker load 命令将这个 tar 文件导入到任何 Docker 环境中…...
虚拟化嵌套
在理论上,可以在虚拟机(VM)内运行一个hypervisor,这个概念被称为嵌套虚拟化: 我们将第一个hypervisor称为Host Hypervisor,将VM内的hypervisor称为Guest Hypervisor。 在Armv8.3-A发布之前,可以通过在EL0中运行Guest Hypervisor来在VM中运行Guest Hypervisor。然而,这…...
【XILINX】记录ISE/Vivado使用过程中遇到的一些warning及解决方案
前言 XILINX/AMD是大家常用的FPGA,但是在使用其开发工具ISE/Vivado时免不了会遇到很多warning,(大家是不是发现程序越大warning越多?),并且还有很多warning根据消除不了,看着特心烦? 我这里汇总一些我遇到的…...
Tableau进阶--Tableau数据故事慧(20)解构Tableau的绘图逻辑
官网介绍 官网连接如下: https://www.tableau.com/zh-cn tableau的产品包括如下: 参考:https://zhuanlan.zhihu.com/p/341882097 Tableau是功能强大、灵活且安全些很高的端到端的数据分析平台,它提供了从数据准备、连接、分析、协作到查阅…...
45.0/HTML 简介(详细版)
目录 45.1 互联网简介 45.2 网页技术与分类 45.3 HTML 简介 45.3.1 什么是 HTML?(面试题) 45.3.2 HTML 文件结构 45.3.3 HTML 语法 45.3.4 实例演练步骤(面试题) 45.4 head 中的常用标签 45.4.1 title 标记 45.4.2 meta 标记 45.4.3 45.4.4 45.4.4(面试题)总结: 45…...
Python 如何进行游戏开发?
游戏开发是一个广泛的领域,Python 作为一门灵活的编程语言,可以用于不同类型的游戏开发。以下是一些建议和步骤,帮助你开始使用 Python 进行游戏开发: 1、选择游戏开发库/框架: Pygame: Pygame 是一个用于…...
到底什么是DevOps
DevOps不是一组工具,也不是一个特定的岗位。在我看来DevOps更像是一种软件开发文化,一种实现快速交付能力的手段。 DevOps 强调的是高效组织团队之间如何通过自动化的工具协作和沟通来完成软件的生命周期管理,从而更快、更频繁地交付更稳定的…...
Keil生成bin文件
Keil生成bin文件_keil5生成bin文件-CSDN博客...
【STM32】USART串口协议
1 通信接口 通信的目的:将一个设备的数据传送到另一个设备,扩展硬件系统 通信协议:制定通信的规则,通信双方按照协议规则进行数据收发 USRT:TX是数据发送引脚,RX是数据接受引脚; I2C…...
淋雨试验箱
产品概述 KDZD-IPX34淋雨试验箱是对户外电子电工产品的防水性能测试的一种装置。该设备通过不同尺寸的喷嘴喷水,产品外壳表面淋水冲洗来检测防水性能。在测试物品时,将样品放在转台上,试验启动时,水流通过压力计和流量计控制水…...
02-MQ入门之RabbitMQ简单概念说明
二:RabbitMQ 介绍 1.RabbitMQ的概念 RabbitMQ 是一个消息中间件:它接受并转发消息。你可以把它当做一个快递站点,当你要发送一个包裹时,你把你的包裹放到快递站,快递员最终会把你的快递送到收件人那里,按…...
敏感信息泄漏怎么破?来试试极狐GitLab 的密钥检测吧
前言 在应用程序开发过程中,一个很常见的问题就是:开发人员为了本地 debug 方便,会 hardcode 一些信息,比如连接数据库的用户名、密码、连接第三方 app 的 token、certificate 等,如果在提交代码的时候没有及时删除 ha…...
go学习之网络编程
文章目录 网络编程1、网络编程的基本介绍2.网络编程的基础知识1)协议(tcp/ip)2)OSI与TCP/ip参考模型3)ip地址4)端口(port)介绍5)tcp socket编程的客户端和服务器端 3.socket编程快速入门4.经典项目-海量用户即时通讯系…...
将数组中的数逆序存放
本题要求编写程序,将给定的n个整数存入数组中,将数组中的这n个数逆序存放,再按顺序输出数组中的元素。 输入格式: 输入在第一行中给出一个正整数n(1≤n≤10)。第二行输入n个整数,用空格分开。 输出格式:…...
Unity Web 浏览器-3D WebView中有关于CanvasWebViewPrefab
一、CanvasWebViewPrefab默认设置 这个是在2_CanvasWebViewDemo示例场景文件中可以可以查看得到,可以看出CanvasWebViewPrefab的默认配置如下。 二、Web 浏览器网页和Unity内置UI的渲染顺序 1、如果你勾选了以下这个Native 2D Mode选项的话,那么Unit…...
一款计算机顶会爬取解析系统 paper info
一款计算机顶会爬取解析系统 paper info 背景项目实现的功能 技术方案架构设计项目使用的技术选型 使用方法本地项目部署使用ChatGPT等大模型创建一个ChatGPT助手使用阿里云 顶会数据量 百度网盘pfd文件json文件 Q&A github链接 :https://github.com/codebricki…...
CommonJs模块化实现原理ES Module模块化原理
CommonJs模块化实现原理 首先看一个案例 初始化项目 npm init npm i webpack -D目录结构如下: webpack.config.js const path require("path"); module.exports {mode: "development",entry: "./src/index.js",output: {path: p…...
实验4.1 静态路由的配置
实验4.1 静态路由的配置 一、任务描述二、任务分析三、具体要求四、实验拓扑五、任务实施1.设置交换机和路由器的基本配置。2.使用display ip interface brief命令查看接口配置信息。3.配置静态路由,实现全网互通。 六、任务验收七、任务小结 一、任务描述 某公司刚…...
postgresql|数据库|只读用户的创建和删除(备忘)
CREATE USER read_only WITH PASSWORD 密码 -- 连接到xxx数据库 \c xxx -- 授予对xxx数据库的只读权限 GRANT CONNECT ON DATABASE xxx TO read_only; GRANT USAGE ON SCHEMA public TO read_only; GRANT SELECT ON ALL TABLES IN SCHEMA public TO read_only; GRANT EXECUTE O…...
sqlserver 根据指定字符 解析拼接字符串
DECLARE LotNo NVARCHAR(50)A,B,C DECLARE xml XML ( SELECT <x> REPLACE(LotNo, ,, </x><x>) </x> ) DECLARE ErrorCode NVARCHAR(50) -- 提取 XML 中的值 SELECT value x.value(., VARCHAR(MAX))…...
【C语言练习】080. 使用C语言实现简单的数据库操作
080. 使用C语言实现简单的数据库操作 080. 使用C语言实现简单的数据库操作使用原生APIODBC接口第三方库ORM框架文件模拟1. 安装SQLite2. 示例代码:使用SQLite创建数据库、表和插入数据3. 编译和运行4. 示例运行输出:5. 注意事项6. 总结080. 使用C语言实现简单的数据库操作 在…...
JUC笔记(上)-复习 涉及死锁 volatile synchronized CAS 原子操作
一、上下文切换 即使单核CPU也可以进行多线程执行代码,CPU会给每个线程分配CPU时间片来实现这个机制。时间片非常短,所以CPU会不断地切换线程执行,从而让我们感觉多个线程是同时执行的。时间片一般是十几毫秒(ms)。通过时间片分配算法执行。…...
SpringTask-03.入门案例
一.入门案例 启动类: package com.sky;import lombok.extern.slf4j.Slf4j; import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; import org.springframework.cache.annotation.EnableCach…...
NPOI Excel用OLE对象的形式插入文件附件以及插入图片
static void Main(string[] args) {XlsWithObjData();Console.WriteLine("输出完成"); }static void XlsWithObjData() {// 创建工作簿和单元格,只有HSSFWorkbook,XSSFWorkbook不可以HSSFWorkbook workbook new HSSFWorkbook();HSSFSheet sheet (HSSFSheet)workboo…...
解析奥地利 XARION激光超声检测系统:无膜光学麦克风 + 无耦合剂的技术协同优势及多元应用
在工业制造领域,无损检测(NDT)的精度与效率直接影响产品质量与生产安全。奥地利 XARION开发的激光超声精密检测系统,以非接触式光学麦克风技术为核心,打破传统检测瓶颈,为半导体、航空航天、汽车制造等行业提供了高灵敏…...
论文阅读笔记——Muffin: Testing Deep Learning Libraries via Neural Architecture Fuzzing
Muffin 论文 现有方法 CRADLE 和 LEMON,依赖模型推理阶段输出进行差分测试,但在训练阶段是不可行的,因为训练阶段直到最后才有固定输出,中间过程是不断变化的。API 库覆盖低,因为各个 API 都是在各种具体场景下使用。…...
前端高频面试题2:浏览器/计算机网络
本专栏相关链接 前端高频面试题1:HTML/CSS 前端高频面试题2:浏览器/计算机网络 前端高频面试题3:JavaScript 1.什么是强缓存、协商缓存? 强缓存: 当浏览器请求资源时,首先检查本地缓存是否命中。如果命…...
java高级——高阶函数、如何定义一个函数式接口类似stream流的filter
java高级——高阶函数、stream流 前情提要文章介绍一、函数伊始1.1 合格的函数1.2 有形的函数2. 函数对象2.1 函数对象——行为参数化2.2 函数对象——延迟执行 二、 函数编程语法1. 函数对象表现形式1.1 Lambda表达式1.2 方法引用(Math::max) 2 函数接口…...
