c# 逆变 / 协变
个人理解:
1. 逆变in向上兼容类
2. 协变out向下兼容类
在面向对象编程中,尤其是使用泛型时,in和out关键字用于限制类型参数的协变性和逆变性。
-
in关键字(逆变):in关键字用于标记泛型类型参数的逆变性。逆变表示可以使用指定类型的基类或超类作为方法的参数类型。- 当一个泛型类或接口使用
in关键字限定类型参数时,该泛型类型可以被赋值给比它更通用的类型。 - 例如,如果有一个只写存储库,你无法通过它查询信息,只能往里面添加动物。这种情况下,可以将泛型类型参数声明为
in Animal,表示该类型参数可以是Animal或其任何基类。
-
out关键字(协变):out关键字用于标记泛型类型参数的协变性。协变表示可以使用指定类型的子类或派生类作为方法的返回类型。- 当一个泛型类或接口使用
out关键字限定类型参数时,该泛型类型可以被赋值给比它更具体的类型。 - 例如,如果有一个只读存储库,你无法通过它添加动物,只能查询信息。这种情况下,可以将泛型类型参数声明为
out Animal,表示该类型参数可以是Animal或其任何派生类。
通过使用in和out关键字,我们可以在泛型类型参数中限制类型的协变性和逆变性,以确保类型安全性。这样做的好处是能够更容易地对泛型类型进行复用和灵活性。
代码示例:
// 声明一个只读存储库接口,用于查询动物信息
interface ReadOnlyRepository<out T> {fun getAll(): List<T>fun getById(id: String): T?
}// 声明一个只写存储库接口,用于添加动物
interface WriteOnlyRepository<in T> {fun add(item: T)
}// Animal类作为基类
open class Animal(val name: String)// Dog类继承自Animal
class Dog(name: String) : Animal(name)// Cat类继承自Animal
class Cat(name: String) : Animal(name)// 只读存储库实现
class ReadOnlyAnimalRepository : ReadOnlyRepository<Animal> {private val animals = listOf(Animal("Lion"), Dog("Buddy"), Cat("Whiskers"))override fun getAll(): List<Animal> {return animals}override fun getById(id: String): Animal? {return animals.find { it.name == id }}
}// 只写存储库实现
class WriteOnlyAnimalRepository : WriteOnlyRepository<Animal> {private val animals = mutableListOf<Animal>()override fun add(item: Animal) {animals.add(item)}
}fun main() {val readOnlyRepo: ReadOnlyRepository<Animal> = ReadOnlyAnimalRepository()val writeOnlyRepo: WriteOnlyRepository<Dog> = WriteOnlyAnimalRepository()val allAnimals = readOnlyRepo.getAll()println("All Animals:")allAnimals.forEach { animal ->println("- ${animal.name}")}val dog1 = Dog("Max")writeOnlyRepo.add(dog1)println("\nAdded Dog:")val addedDog = readOnlyRepo.getById(dog1.name)println("- ${addedDog?.name}")val cat1 = Cat("Misty")// writeOnlyRepo.add(cat1) // Compilation Error: Type mismatch. Required: Dog, Found: Cat
}
在上面的代码示例中,我们定义了两个存储库接口:ReadOnlyRepository和WriteOnlyRepository。ReadOnlyRepository接口声明了返回类型out T,表示它只能查询(读取)动物信息。WriteOnlyRepository接口声明了参数类型in T,表示它只能添加(写入)动物。
然后,我们创建了一个ReadOnlyAnimalRepository类来实现ReadOnlyRepository<Animal>接口,用于查询动物信息。同样地,我们还创建了一个WriteOnlyAnimalRepository类来实现WriteOnlyRepository<Animal>接口,用于添加动物。
在main函数中,我们首先将ReadOnlyAnimalRepository赋值给readOnlyRepo变量,这是合法的,因为ReadOnlyAnimalRepository实现了ReadOnlyRepository<Animal>接口。
然后,我们将WriteOnlyAnimalRepository赋值给writeOnlyRepo变量,这也是合法的,因为WriteOnlyAnimalRepository实现了WriteOnlyRepository<Animal>接口。注意,虽然我们将其赋值给writeOnlyRepo变量,但我们仍然只能使用add方法添加
相关文章:
c# 逆变 / 协变
个人理解: 1. 逆变in向上兼容类 2. 协变out向下兼容类 在面向对象编程中,尤其是使用泛型时,in和out关键字用于限制类型参数的协变性和逆变性。 in关键字(逆变): in关键字用于标记泛型类型参数的逆变性。…...
electron使用better-sqlite3打包失败(electron打包有进程没有界面)
remove *\chrome_100_percent.pak: Access is denied. 解决: 管理员权限执行:taskkill /IM 你的进程名.exe /F,再次执行build electron使用better-sqlite3打包后有进程没有界面 原因是代码及依赖包安装有误,模块丢失。主要分享的…...
2.6文件服务器
2.6文件服务器 一、Ftp 介绍 文件传输协议(File Transfer Protocol,FTP),基于该协议FTP客户端与服务端可以实现共享文 件、上传文件、下载文件。 FTP 基于TCP协议生成一个虚拟的连接,主要用于控制FTP连接信息&#x…...
【C++ 学习 ㊴】- 详解 C++ 的 I/O 流
目录 一、C 的 I/O 流 二、C 的标准 I/O 流 三、C 的文件 I/O 流 一、C 的 I/O 流 C 语言有一套完成数据读写(I/O)的解决方案: 使用 scanf()、gets() 等函数从键盘读取数据,使用 printf()、puts() 等函数向屏幕输出数据&#…...
js算法面试题(附答案)
js算法面试题十道 两数之和 题目:给定一个整数数组 nums 和一个目标值 target,请你在该数组中找出和为目标值的那两个整数,并返回他们的数组下标。 function twoSum(nums, target) {const map new Map();for (let i 0; i < nums.leng…...
2023 年戴森设计大奖得主是谁?给大楼降温、争取救援机会
2023 年戴森设计大奖得主是谁?给大楼降温、争取救援机会 编辑拉风的极客2023/11/22 摘要 当今社会除了持续不断对科技创新保持注目,还有很多年轻发明家为了实际场景的难题提供解决方案。 11 月 15 日,2023 年戴森设计大奖国际大奖名单正…...
〖大前端 - 基础入门三大核心之JS篇㊲〗- DOM改变元素节点的css样式、HTML属性
说明:该文属于 大前端全栈架构白宝书专栏,目前阶段免费,如需要项目实战或者是体系化资源,文末名片加V!作者:不渴望力量的哈士奇(哈哥),十余年工作经验, 从事过全栈研发、产品经理等工作…...
【word技巧】Word制作试卷,ABCD选项如何对齐?
使用word文件制作试卷,如何将ABCD选项全部设置对齐?除了一直按空格或者Tab键以外,还有其他方法吗?今天分享如何将ABCD选项对齐。 首先,我们打开【替换和查找】,在查找内容输入空格,然后点击全部…...
OpenHarmony 4.1计划明年Q1发布, 5.0预计Q3发布
据HarmonyOS官方组织透露,OpenHarmony 4.0 版本已于 10 月 26 日正式发布,开发套件同步升级到 API 10。开放原子开源基金会现更新了 OpenHarmony 4.1&5.0 版本路线图。据介绍,OpenHarmony 4.1 Beta 版本预计将于年底完成测试并发布&#…...
蓝桥等考C++组别八级002
第一部分:选择题 1、C++ L8 (15分) 整数12,8的最小公倍数是( )。 A. 4 B. 16 C. 24 D. 48 正确答案:C 2、C+&#...
秋招JAVA面经总结
面试的范围是Java基础+Java并发+Java框架+mysql+网络。 Java基础 重载与重写有什么区别? 重载(Overloading)指的是在同一个类中,可以有多个同名方法,它们具有不同的参数列表(参数类型、参数个数或参数顺序不同),编译器根据调用时的参数类型来决定调用哪个方法。 重写…...
Postgresql源码(116)提升子查询案例分析
0 总结 对于SQL:select * from student, (select * from score where sno > 2) s where student.sno s.sno; pullup在pull_up_subqueries函数内递归完成,分几步: 将内层rte score追加到上层rtbable中:rte1是student、rte2带…...
CNP实现应用CD部署
上一篇整体介绍了cnp的功能,这篇重点介绍下CNP产品应用开发的功能。 简介 CNP的应用开发,主要是指的应用CD部署的配置管理。 应用列表,用来创建一个应用,一般与项目对应,也可以多个应用对应到一个项目。具体很灵活。…...
kubeadm join 192.168.10.16:6443 --token xxx报错Failed to request cluster-info
1、node节点执行 kubeadm join 192.168.10.16:6443 --token hak4zi.hrib9uv4p62t1uok --discovery-token-ca-cert-hash sha256:4337638eef783ee6a66045ad699722079e071c2dfbaa21e37d3174f04d58ea97 --v2 报错 [discovery] Failed to request cluster-info, will try again: G…...
车载以太网-传输层-TCP
文章目录 TCP协议TCP协议报文格式TCP报文的示例TCP建立连接TCP断开连接TCP协议测试TCP协议 车载以太网TCP协议是一种在车载以太网网络中使用的传输控制协议(TCP)。它是一种面向连接的协议,用于在车辆之间或车辆与基础设施之间传输数据。TCP协议提供了可靠的数据传输,确保数…...
java:简单入门定时任务的几种方式Timer、Quartz、Spring Task
背景 后端的定时任务在许多应用中都扮演着重要的角色,它们可用于处理重复性任务、执行定期操作或处理需要定时触发的任务。以下是一些使用场景的示例: 数据同步:在分布式系统中,不同系统之间经常需要进行数据同步。例如…...
木子-前端-方法标签属性小记(普通jsp/html篇)2023~2024
目录 1、如何在前端页面将base转成存进input的图片格式 2、通过前端页面判断当前使用方式PC端还是手机端的极简易方法 1、如何在前端页面将base转成存进input的图片格式 不是直接存进input里,只是将文件转成对应的格式。 data.picPath是base64编码,我…...
音视频项目—基于FFmpeg和SDL的音视频播放器解析(十七)
介绍 在本系列,我打算花大篇幅讲解我的 gitee 项目音视频播放器,在这个项目,您可以学到音视频解封装,解码,SDL渲染相关的知识。您对源代码感兴趣的话,请查看基于FFmpeg和SDL的音视频播放器 如果您不理解本…...
使用netty实现WebSocket协议通信
服务器与浏览器之间实现通信,一般都是由浏览器发起http请求,服务端对http请求进行响应,要实现服务端主动向浏览器推送数据,一般采用的方案都是websocket主动推送,或者前端实现轮询方式拉取数据,轮询方式多少…...
uniapp开发小程序,包过大解决方案
1、首先和大家说一下 微信小程序 主包限制不能超过2M 分包一共不能超过8M 然后具体解决优化步骤如下, 将主包进行分包 在pages.json 下subPackages里面进行配置分包 分包配置完 配置过的文件都需要进行修改对应的路径 2 、 在运行的时候 一定要勾选 压缩代码 有…...
语音助手DNS查询技能开发:从API集成到智能家居运维实践
1. 项目概述:一个DNS查询技能的诞生最近在折腾智能音箱和家庭自动化,发现一个挺有意思的需求:能不能直接问家里的智能设备,某个网站的DNS解析情况?比如,想知道“我的博客域名解析生效了吗?”或者…...
3个关键步骤掌握Cellpose:如何实现超越人工的细胞分割精度?
3个关键步骤掌握Cellpose:如何实现超越人工的细胞分割精度? 【免费下载链接】cellpose a generalist algorithm for cellular segmentation with human-in-the-loop capabilities 项目地址: https://gitcode.com/gh_mirrors/ce/cellpose Cellpose…...
利润下滑39%!瑞幸的“万店神话”还灵吗?
在热闹的五一节日以前,瑞幸咖啡于4月就早早开始了它的劳动:宣布进军即饮咖啡市场。在外部媒体看来,瑞幸这一动作的主要目的,是为了从“门店咖啡”向“全场景咖啡”转型。这一点,从它签下顶流明星王一博作为全球品牌代言…...
采购成本总是说不清?怎么打造让老板一眼看透的采购成本地图?
每100元营收中,就有超过60元经由采购之手花出去。 然而,辛辛苦苦一年,到了年底,采购最常听到的却是…… “钱到底花哪了?” “还能再降降吗?” 只有采购人员知道,各项费用成本明细分散在纸质合同…...
高性能Android容器化方案:Waydroid在Linux上的架构解析与部署指南
高性能Android容器化方案:Waydroid在Linux上的架构解析与部署指南 【免费下载链接】waydroid Waydroid uses a container-based approach to boot a full Android system on a regular GNU/Linux system like Ubuntu. 项目地址: https://gitcode.com/gh_mirrors/w…...
ASMR资源管理新范式:asmroner如何重新定义音频内容获取体验
ASMR资源管理新范式:asmroner如何重新定义音频内容获取体验 【免费下载链接】asmr-downloader A tool for download asmr media from asmr.one(Thanks for the asmr.one) 项目地址: https://gitcode.com/gh_mirrors/as/asmr-downloader 你是否曾为寻找高质量…...
3步完成!Media Extended Bilibili插件完整安装配置指南
3步完成!Media Extended Bilibili插件完整安装配置指南 【免费下载链接】mx-bili-plugin 项目地址: https://gitcode.com/gh_mirrors/mx/mx-bili-plugin 想要在Obsidian笔记中无缝播放Bilibili视频吗?Media Extended Bilibili插件为你提供了强大…...
想快速导出视频字幕?2026年剪映导出字幕文字的方法+提词匠全能方案
如果你是内容创作者、学生或办公族,经常需要从视频里提取文案或生成字幕文件,会发现卡在这几个地方:有时候视频自带的字幕格式不好导,有时候视频根本没字幕需要自己识别,有时候识别出来的文本还要手工修改时间戳。这篇…...
告别模拟器:Windows上直接运行APK的终极解决方案
告别模拟器:Windows上直接运行APK的终极解决方案 【免费下载链接】APK-Installer An Android Application Installer for Windows 项目地址: https://gitcode.com/GitHub_Trending/ap/APK-Installer 想在Windows电脑上直接运行安卓应用,不再需要笨…...
【AISMM模型落地实战白皮书】:20年架构师亲授5大避坑指南与3个行业成功范式
更多请点击: https://intelliparadigm.com 第一章:AISMM模型在企业落地实践指南 AISMM(AI-Driven Secure Maturity Model)是一套面向AI系统全生命周期的安全成熟度评估与演进框架,强调可度量、可审计、可迭代的工程化…...
