Swift 中 associatedtype 的用法详解

目录

前言

1.什么是associatedtype

2.associatedtype 的作用

1.让协议支持泛型

2.让协议支持不同的数据类型

3.结合 where 关键字限制类型

4.什么时候使用 associatedtype

5.总结


前言

        在 Swift 语言中,泛型(Generics)是一个非常强大的特性,它允许我们编写灵活且可复用的代码。而当我们在 协议(Protocol) 中需要使用泛型时,associatedtype 就派上了用场。

        本文将详细介绍 associatedtype 的作用、使用场景,并通过代码示例帮助大家更好地理解它的使用方式。

1.什么是associatedtype

        在 Swift 的协议中,我们无法直接使用泛型 <T>,但可以使用 associatedtype 关键字来声明一个占位类型,让协议在不确定具体类型的情况下仍然能够正常使用。

        associiatedtype的语法如下:

protocol SomeProtocol {associatedtype SomeTypefunc doSomething(with value: SomeType)
}
  1. associatedtype SomeType:声明一个占位类型 SomeType,但不指定具体类型。
  2. func doSomething(with value: SomeType):SomeType 由实现该协议的类型决定。

        这样,任何遵循 SomeProtocol 的类型都可以自定义 SomeType 为任何符合需求的类型,从而提高协议的通用性。

2.associatedtype 的作用

1.让协议支持泛型

        假设我们想设计一个容器(Container)协议,让它能够存储不同类型的元素,如果不用 associatedtype,我们可能会写成:

protocol Container {func append(_ item: Int)func getItem(at index: Int) -> Int
}

        这个协议只能存储 Int 类型的元素,缺乏灵活性。

        而使用 associatedtype,可以让它支持任意类型:

protocol Container {associatedtype Itemfunc append(_ item: Item)func getItem(at index: Int) -> Item
}

        现在,我们可以创建不同类型的容器:

struct IntContainer: Container {typealias Item = Int  // 指定 Item 为 Int 类型private var items: [Int] = []func append(_ item: Int) {items.append(item)}func getItem(at index: Int) -> Int {return items[index]}
}struct StringContainer: Container {typealias Item = String  // 指定 Item 为 String 类型private var items: [String] = []func append(_ item: String) {items.append(item)}func getItem(at index: Int) -> String {return items[index]}
}

        这样 IntContainer 和 StringContainer 都遵循 Container 协议,但它们的 Item 类型不同,提高了代码的通用性。

2.让协议支持不同的数据类型

        假设我们要设计一个**栈(Stack)**数据结构,它应该支持不同的数据类型,比如 Int、String 等:

protocol StackProtocol {associatedtype Elementmutating func push(_ item: Element)mutating func pop() -> Element?
}

        不同的数据类型可以实现这个协议:

struct IntStack: StackProtocol {typealias Element = Intprivate var stack: [Int] = []mutating func push(_ item: Int) {stack.append(item)}mutating func pop() -> Int? {return stack.popLast()}
}struct StringStack: StackProtocol {typealias Element = Stringprivate var stack: [String] = []mutating func push(_ item: String) {stack.append(item)}mutating func pop() -> String? {return stack.popLast()}
}

        IntStack 和 StringStack 都遵循 StackProtocol,但 Element 类型可以不同!

3.结合 where 关键字限制类型

        有时候,我们希望 associatedtype 只能是某种类型的子类或实现了某个协议。可以使用 where 关键字进行类型约束:

protocol Summable {associatedtype Number: Numeric  // 限定 Number 必须是 Numeric 协议的子类型func sum(a: Number, b: Number) -> Number
}

        实现 Summable 协议:

struct IntegerAdder: Summable {func sum(a: Int, b: Int) -> Int {return a + b}
}struct DoubleAdder: Summable {func sum(a: Double, b: Double) -> Double {return a + b}
}

        这里 Number 只能是 Int、Double 之类的 Numeric 类型,保证了类型安全。

3.associatedtype 与泛型的区别

比较项

associatedtype(协议中的泛型)

普通泛型 <T>

适用范围

只能用于 协议

可用于 类、结构体、函数

作用

让协议支持不确定的类型,由实现者决定具体类型

让类型/函数支持泛型

例子

protocol Container { associatedtype Item }

struct Stack<T> {}

限制

只能用于协议,不能直接实例化

适用于所有类型

4.什么时候使用 associatedtype

       当你需要创建一个通用的协议,但不想限定某个具体类型时。

        当不同的实现类需要指定不同的数据类型时。

        当你希望协议中的某些类型参数具备类型约束时(如 where 关键字)。

5.总结

        associatedtype允许协议拥有不确定的类型,由实现者决定具体类型。

        让协议支持泛型,使其更加通用,适用于不同的数据类型。

        可以通过where限制类型范围,提高安全性。

        适用于 协议,而泛型 <T> 适用于类、结构体和函数。

        一句话总结:associatedtype 就是协议的泛型,让协议更加灵活和可扩展!

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

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

相关文章

每日Attention学习26——Dynamic Weighted Feature Fusion

模块出处 [ACM MM 23] [link] [code] Efficient Parallel Multi-Scale Detail and Semantic Encoding Network for Lightweight Semantic Segmentation 模块名称 Dynamic Weighted Feature Fusion (DWFF) 模块作用 双级特征融合 模块结构 模块思想 我们提出了 DWFF 策略&am…

OpenCV实现图像特征提取与匹配

‌一、特征检测与描述子提取‌ ‌选择特征检测器‌ 常用算法包括&#xff1a; ‌ORB‌&#xff1a;一种高效的替代SIFT和SURF的算法&#xff0c;主要用于移动机器人和增强现实等领域。适合实时应用&#xff0c;结合FAST关键点与BRIEF描述子‌。‌SIFT&#xff08;尺度不变特征变…

向量检索在AI中的应用与技术解析

关键要点 向量检索在AI中用于信息检索、推荐系统和图像搜索&#xff0c;研究表明其通过高维空间中的向量表示数据来提升搜索相关性。它依赖于嵌入技术&#xff08;如Word2Vec、BERT&#xff09;和近邻算法&#xff08;如kNN、ANN&#xff09;&#xff0c;证据倾向于其在处理大…

事务与异步方法(@Async)协同工作

目录 1. 问题场景与风险 &#xff08;1&#xff09;典型场景 &#xff08;2&#xff09;风险分析 2. 解决方案&#xff1a;事务提交后触发异步操作 &#xff08;1&#xff09;代码示例 &#xff08;2&#xff09;关键注解 3. 原理解析 &#xff08;1&#xff09;事务同…

关于进程的实验(子进程和父进程相关的)

文章目录 1.第一个问题2.第二个问题3.第三个问题 1.第一个问题 编写一段程序&#xff0c;利用系统调用fork( )创建两个进程。当此程序运行时&#xff0c;在系统中有一个父进程和两个子进程活动。让每一个进程在屏幕上显示一个字符&#xff1a;父进程显示字符“a”;子进程分别显…

MyBatis 如何创建 SqlSession 对象的?

MyBatis 创建 SqlSession 对象的过程主要由 SqlSessionFactory 接口及其实现类来完成。以下是详细步骤&#xff1a; 1. SqlSessionFactory 接口: SqlSessionFactory 是 MyBatis 的核心接口之一&#xff0c;它负责创建 SqlSession 对象。 你可以将 SqlSessionFactory 视为 Sql…

深度优先搜索(DFS)剪枝技术详解与C++实现

深度优先搜索&#xff08;DFS&#xff09;剪枝技术通过提前终止无效路径的搜索&#xff0c;大幅提升算法效率。以下是五种核心剪枝技术的详细解析及C代码示例&#xff1a; 目录 一、可行性剪枝 C实现示例 二、搜索顺序剪枝 伪代码逻辑 三、最优性剪枝 C实现示例 四、排除…

【双指针】移动零

题目描述&#xff1a; 算法分析&#xff1a; 观察输入输出&#xff1a; 输出中一共分为两个区域&#xff0c;0区和非零区。 但是在处理未完成之前&#xff0c;必然存在着一个零和非零数共存的区域&#xff0c;所以在处理的过程当中一共有三个区域&#xff0c;0区&#xff0c;…

学习15天:pytest

1、.pytest强大的插件 pytest-html(生成html格式的自动化测试报告) pytest-xdist测试用例分布式执行。多CPU分发。 pytest-ordering 用于改变测试用例的执行顺序 pytest-rerunfailures用例失败后重跑 allure-pytest 用于生成美观的测试报告。 2、规则&#xff1a; 模块…

股票交易所官方api接口有哪些?获取和使用需要满足什么条件

炒股自动化&#xff1a;申请官方API接口&#xff0c;散户也可以 python炒股自动化&#xff08;0&#xff09;&#xff0c;申请券商API接口 python炒股自动化&#xff08;1&#xff09;&#xff0c;量化交易接口区别 Python炒股自动化&#xff08;2&#xff09;&#xff1a;获取…

2.7 滑动窗口专题:串联所有单词的子串

LeetCode 30. 串联所有单词的子串算法对比分析 1. 题目链接 LeetCode 30. 串联所有单词的子串 2. 题目描述 给定一个字符串 s 和一个字符串数组 words&#xff0c;words 中所有单词长度相同。要求找到 s 中所有起始索引&#xff0c;使得从该位置开始的连续子串包含 words 中所…

【区块链】区块链密码学基础

&#x1f308;个人主页: 鑫宝Code &#x1f525;热门专栏: 闲话杂谈&#xff5c; 炫酷HTML | JavaScript基础 ​&#x1f4ab;个人格言: "如无必要&#xff0c;勿增实体" 文章目录 区块链密码学基础引言一、哈希函数1.1 基本概念1.2 数学表达 二、非对称加密2.1…

Spring Boot配置类原理、Spring Boot核心机制理解,以及实现自动装置的底层原理

目的:从底层源码角度分析 Spring Boot 配置类以及自动装载的底层原理 文章目录 1. Spring Boot 配置类实现自动装载1.1 @Configuration注解1.2 @Configuration 注解完成 bean 注入流程图1.3 @ConfigurationProperties注解赋值2. Spring Boot的核心机制:自动装配2.1 @SpringBo…

docker桌面版启动redis,解决无法连接

docker run -d --name redis -p 6379:6379 -v E:\2\redis\redis.conf:/usr/local/etc/redis/redis.conf redis redis-server /usr/local/etc/redis/redis.conf 在本地创建一个目录&#xff0c;里面有个redis.conf文件&#xff0c;内容如下&#xff0c;启动时绑定这个配置文件目…

[网络][tcp协议]:tcp报头

tcp(传输控制协议)是一种面向字节流的传输层协议,相较于udp协议,tcp能保证传输数据的可靠性与准确性,tcp也是目前最常见的传输层协议 本文主要介绍tcp报头各个字段的含义与用途 注:保留6位和6位标记位是目前最普遍的写法,在我查资料时,发现有一些拓展情况,会在后文细说 最简单的…

【虚幻C++笔记】引擎源码下载及编译步骤

目录 1.在GitHub上访问虚幻引擎源代码2.安装Visual Studio 20223.解压完成以后&#xff0c;打开源码的根目录&#xff0c;选择Setup.bat运行4.选择GenerateProjectFiles.bat运行,生成uE5.sln文件&#xff0c;点击这个文件打开项目5.设置编译的选项&#xff0c;选择DevelopmentE…

【数学建模】层次分析法(AHP)详解及其应用

层次分析法(AHP)详解及其应用 引言 在现实生活和工作中&#xff0c;我们经常面临复杂的决策问题&#xff0c;这些问题通常涉及多个评价准则&#xff0c;且各准则之间可能存在相互影响。如何在这些复杂因素中做出合理的决策&#xff1f;层次分析法(Analytic Hierarchy Process…

科普:为何要对特征进行分箱?

一、为何要对特征进行分箱&#xff1f; 分箱&#xff08;Binning&#xff09;是将连续型或离散型特征转化为区间型变量的过程&#xff0c;其核心目标是提升模型效果和解释性&#xff0c;具体原因如下&#xff1a; 1. 业务需求 可解释性&#xff1a;将特征转化为业务可理解的…

理解langgraph工作流的驱动逻辑,以适应langgraph工作流模式的编程。

langgraph的工作流模式虽然方便直观&#xff0c;但习惯了普通函数式编程的数据流处理。刚开始接触时&#xff0c;确实容易试图用函数式编程的思维去适配它&#xff0c;特别是langgraph数据传递由状态字典管理&#xff0c;而非函数返回值&#xff0c;导致代码不够自然&#xff0…

线性dp(数字三角形,LIS,LCS,LCIS)

文章目录 线性dp数字三角形题目思路 LIS&#xff08;最长上升子序列&#xff09;代码&#xff08;n^2&#xff09;二分优化&#xff08;nlogn&#xff09; LCS(最长公共子序列)代码 LCS——>>LIS思路代码 最长公共子串最长公共上升子序列&#xff08;LCIS&#xff09; 线…