Swift 初学者交心:在 Array 和 Set 之间我们该如何抉择?

概述
初学 Swift 且头发茂密的小码农们在日常开发中必定会在数组(Array)和集合(Set)两种类型之间的选择中“摇摆不定”,这也是人之常情。

Array 和 Set 在某些方面“亲如兄弟”,但实际上它们之间却有着“云泥之别”。
在本篇博文中,您将学到如下内容:
- 概述
- 1. Array 其人
- 2. Set 其人
- 3. 考量运行性能
- 4. Array 和 Set 选择原则
- 总结
相信学完本课后,小伙伴们定能在开发中对如何选择 Array 和 Set 两种数据类型胸有成竹。
闲言少叙,Let’s make a choice!!!😉
1. Array 其人
Array 俗称数组,它其实上就是一个遵守 Collection 协议的泛型结构。在 Swift 中我们有很多种定义它的方法:
var ary0 = Array<Int>()
var ary1 = [Int]()
var ary2: [Int] = []let ary3 = [Int](repeating: 0, count: 100)
let ary4 = [1,2,3,4,5]
let ary5 = Array((0..<100))
如上代码所示,我们还可以通过 let 和 var 关键字来调整数组的可变性。
一般来说,Swift 数组中的元素类型都是同构(或同质,homogeneous)的,这可以从它定义中只有一个泛型元素类型参数来得到印证。

这里不考虑使用 AnyObject 或 Any 作为数组的元素类型,也不考虑使用 Swift 5.7 之后的 any ProtocolType 来“打包存放”遵循相同协议的不同元素类型。
不过通过巧妙的利用枚举,我们可以间接的向其内部“注入”异构(heterogeneous)类型的数据:
enum MixedValue {case int(Int)case string(String)
}let ary: [MixedValue] = [.int(1337), .string("Hello")]
如上代码所示:虽然表面上看数组 ary 中放入了 Int 和 String 两种不同类型的数据,但其实它们都是同一种 MixedValue 枚举类型。
关于更多 Swift 中枚举类型的介绍,请小伙伴们恣意观赏下面的视频课:
Swift 基本功修炼:深入浅出 Swift 中的枚举类型
Array 中元素的存放是有顺序的,无论我们遍历数组 1 千遍或是 1 万遍它们的顺序都是绝对不会变的。
除此之外,数组中的元素是可以相同的,没有哪条规定说数组不能放入一样的东西进去。这也是为什么数组定义中其元素(Element)类型是没有任何限制条件的。
Array 本身为我们提供了丰富的方法可供调用,其中最常用的一个恐怕得算是 first(where:) 方法了,它被用来在数组中查找符合条件的元素:
let ary: [Int] = [1337, 1338, 1339]let item = ary.first(where: { $0 == 1340 })
需要注意的是:first(where:) 方法可能返回 nil 值哦。
2. Set 其人
简单聊完了数组之后,接下来我们再来看看 Set 类型。
Set 同样是一种保存同质元素的容器,我们也可以用多种方式来初始化它的实例:
let set0 = Set<Int> ([1, 2, 2, 3, 3, 4, 4, 5, 5])
print (set0)var set1: Set<String> = []
set1.insert ("Hopy")
set1.insert ("Hopy")
set1.insert ("Panda" )
print (set1)
从上面代码来看 Set 和 Array 很相似,不过它们却有一个关键的不同:Set 中的元素必须遵守 Hashable 协议。

这意味着:不遵循 Hashable 的类型不能放到 Set 中去,这在我们创建自定义类型时尤其需要注意。
遵循 Hashable 协议带来的另一个重要限制是:Set 中的元素必须拥有相同的 Hash 值才行。如果插入 Set 的新元素和 Set 已有元素的 Hash 值相同,则老元素会保持不动,新元素则会被无情的“拒绝”。
class Item: Hashable, Equatable {let value: Intlet name: Stringstatic func ==(lhs: Item, rhs: Item) -> Bool {lhs.value == rhs.value}func hash(into hasher: inout Hasher) {hasher.combine(value)}init(value: Int, name: String) {self.value = valueself.name = name}deinit {print("\(name) deinited!")}
}var mySet = Set<Item>()func test() {let item_0 = Item(value: 1, name: "hopy")let item_1 = Item(value: 1, name: "大熊猫侯佩")mySet.insert(item_0)mySet.insert(item_1)
}test()
print(mySet.first?.name)
以上代码运行结果如下所示:

我们可以用 contains() 方法来检查 Set 是否包含指定的元素,我们同样可以用与数组类似的 first(where:) 方法来查找集合中符合条件的元素。
Set 和数组另一个非常重要的区别是:Set 是无序的。
当我们多次遍历同一个 Set 时即可发现:其内部元素的顺序是无法保持稳定的。
如果大家希望兼顾 Array 和 Set 两者的优点:有序且唯一,我们可以使用 Apple 官方 swift-collections package 库中的 OrderedSet 类型:
- apple / swift-collections

更多 Swift 语言中集合类型的介绍,请感兴趣的小伙伴们移步如下博文观赏进一步精彩的内容:
- Swift 入门学习:集合(Collection)类型趣谈-上
- Swift 入门学习:集合(Collection)类型趣谈-下
3. 考量运行性能
现在,我们已经分别简单介绍了 Swift 中的 Array 和 Set 两种类型,不过它们到底“孰是孰非”,我们还是无所适从。
接下来我们就从两者运行性能的角度来分析一番吧。
分析数组和集合性能的重要因素是它们内部元素遍历的速度。
数组元素遍历性能是 O(n),这意味着在最坏的情况下我们需要找遍数组中的所有元素才能得偿所愿,这在数组包含海量元素时会变得“蜗行牛步”。
反过来集合元素遍历性能却是 O(1),这说明要捕获 Set 内部任意元素所花费的时间都是一个“常数”,不管它们的数量是 100 个或是 1 亿个。
通过上面的讨论,我们清楚可知一点:如果我们的目的就是用最快的速度从一大堆对象里查找心仪的元素,那么 Set 是当仁不让的选择。
包含海量元素的数组在使用中往往是性能的“阿格琉斯之踵”,如果对此感兴趣的小伙伴们可以移步到以下博文中进一步学习性能优化之道:
- 你敢信!?几行代码让Swift数组初始化提速400多倍!
- SwiftUI一招让List巨量数据刷新UI速度快100+倍
- 有趣简单的M2处理器性能实验:Swift与C代码执行速度的比较
- Swift忽略大小写搜索子字符串的三种方法及性能对比
4. Array 和 Set 选择原则
综上所述,我们认为在 Set 和 Array 之间选择的最佳决策是基于实际撸码的“情境”。
当我们拥有一个哈希项目(Item)列表,这些项目在一个集合中需要是唯一的,且无需排序,这时我们倾向于使用 Set。
如果你关心秩序,或者你无法使 Item 遵守 Hashable 协议(这貌似不大可能),那么我们更应考虑使用数组。
当然,也有一个例外:即我们可能希望在维持元素顺序稳定时同时保证唯一性。在这种情况下,我们可以选择使用 swift-collections 框架中的OrderedSet(有序集合)类型。
最后一个非常重要的撸码要诀是:我们的决定总是优先基于以上原则,而不是首先考虑性能之类的东东。除非我们正在编写一段性能非常关键的代码,而在该代码中有足够的证据来证明 Set 和 Array 间的性能优劣。
记住:过早优化是万恶之源!切记!切记!
想要系统学习 Swift 语言的小伙伴们,千万不要错过我的《Swift 语言开发精讲》专栏哦,欢迎大家恣意观赏:

- Swift 语言开发精讲
总结
在本篇博文中,我们简单介绍了 Swift 语言中数组(Array)和集合(Set)两种类型的特性,并比较了它们在实际运行中的性能,最后我们进一步探讨了选择它们的基本原则。
感谢观赏,再会!😎
相关文章:
Swift 初学者交心:在 Array 和 Set 之间我们该如何抉择?
概述 初学 Swift 且头发茂密的小码农们在日常开发中必定会在数组(Array)和集合(Set)两种类型之间的选择中“摇摆不定”,这也是人之常情。 Array 和 Set 在某些方面“亲如兄弟”,但实际上它们之间却有着“云…...
C++ 类模板 函数模板
类模板 #include <bits/stdc.h> using namespace std; //多少变量就写多少个 template<typename T1, typename T2> class Cat { public:Cat(){}Cat(T1 name, T2 age){this->age age;this->name name;}void print(){cout << this->name << …...
OTP8脚-全自动擦鞋机WTN6020-低成本语音方案
一,产品开发背景 首先,随着人们生活质量的提升,对鞋子的保养需求也日益增加。鞋子作为人们日常穿着的重要组成部分,其清洁度和外观状态直接影响到个人形象和舒适度。因此,一种能够自动清洁和擦亮鞋子的设备应运而生&am…...
GpuMall智算云:meta-llama/llama3/Llama3-8B-Instruct-WebUI
LLaMA 模型的第三代,是 LLaMA 2 的一个更大和更强的版本。LLaMA 3 拥有 35 亿个参数,训练在更大的文本数据集上GpuMall智算云 | 省钱、好用、弹性。租GPU就上GpuMall,面向AI开发者的GPU云平台 Llama 3 的推出标志着 Meta 基于 Llama 2 架构推出了四个新…...
内存泄漏案例分享4-异步任务流内存泄漏
案例4——异步任务内存泄漏 异步任务,代指起子线程异步完成一些数据操作、网络接口请求等,通常会使用以下API: Runnbale,Thread,线程池RxJavaHandlerThread 而这些异步任务很有可能操作内存泄漏,下面我们以Rxjava为…...
【机器学习300问】100、怎么理解卷积神经网络CNN中的池化操作?
一、什么是池化? 卷积神经网络(CNN)中的池化(Pooling)操作是一种下采样技术,其目的是减少数据的空间维度(宽度和高度),同时保持最重要的特征并降低计算复杂度。池化操作不…...
RPA机器人流程自动化如何优化人力资源工作流程
人力资源部门在支持员工和改善整体工作环节方面扮演着至关重要的角色,但是在人资管理的日常工作中,充斥着大量基于规则的重复性任务,例如简历筛选、面试安排、员工数据管理、培训管理、绩效管理等,这些任务通常需要工作人员花费大…...
OpenHarmony开发者大会2024:鸿心聚力 智引未来
2024年5月25日,OpenAtom OpenHarmony(简称“OpenHarmony")委员会以“鸿心聚力,智引未来”为主题,在创新之城深圳举办OpenHarmony开发者大会2024,为开发者、产业组织、生态伙伴和行业客户搭建一个交流、分享和学习…...
新楚文化知网收录文学艺术类期刊投稿
《新楚文化》是由国家新闻出版总署批准,湖北省文学艺术界联合会主管,湖北今古传奇传媒集团有限公司主办的正规期刊。主要刊登文化、文学、艺术类稿件;包括传统文化、非遗、历史文化、地方文化、中外友好文化交流、文学作品研究、艺术研究等方…...
基于vue3速学angular
因为工作原因,需要接手新的项目,新的项目是angular框架的,自学下和vue3的区别,写篇博客记录下: 参考:https://zhuanlan.zhihu.com/p/546843290?utm_id0 1.结构上: vue3:一个vue文件ÿ…...
链游中的代币(Token)或加密货币(Cryptocurrency)是如何产生和使用的?
在区块链游戏(链游)中,代币和加密货币不仅是游戏经济的核心,也是连接现实世界与虚拟游戏世界的桥梁。这些数字货币不仅赋予了游戏内资产的真实价值,还为玩家提供了全新的互动和交易方式。下面,我们将深入探…...
2024年5月23日 (周四) 叶子游戏新闻
《Unclogged》Steam页面上线 马桶主题恐怖逃脱解谜Brody制作并发行,一款奇葩创意马桶主题恐怖逃脱解谜新游《Unclogged》Steam页面上线,本作暂不支持中文。 Meta人工智能主管杨立昆 大语言模型不会达到人类智能水平IT之家今日(5月23日&#x…...
猫毛过敏终结者!宠物空气净化器让你告别红眼和喷嚏
猫毛过敏是一种常见的过敏性疾病,影响着全球数百万人的日常生活。这种过敏反应通常是由于对猫皮屑、唾液或尿液中的蛋白质产生免疫反应而引起的。症状可能包括打喷嚏、流鼻涕、眼睛痒、皮肤疹和呼吸困难,严重影响患者的舒适度和生活质量。对于猫毛过敏者…...
xgboost项目实战-保险赔偿额预测与信用卡评分预测001
目录 算法代码 原理 算法流程 xgb.train中的参数介绍 params min_child_weight gamma 技巧 算法代码 代码获取方式:链接:https://pan.baidu.com/s/1QV7nMC5ds5wSh-M9kuiwew?pwdx48l 提取码:x48l 特征直方图统计: fig, …...
子网划分,交换机原理与配置
子网划分 IP地址 IPv4由32位二进制数组成,一般用点分十进制来表示 IPv4是由32位二进制数组成,分成四组,第组八位。例如:11000000.10101000.00000000.00000010 为了便于配置通常表示成点分十进制形式例如:192.168.0.2 255.255.255.0 IPv6由128位组成&…...
记mapboxGL实现鼠标经过高亮时的一个问题
概述 mapboxGL实现鼠标经过高亮可通过注册图层的mousemove和moveout事件来实现,在mousemove事件中可以拿到当前经过的要素,但是当使用该要素时,发现在某个地图级别下会有线和面数据展示不全的情况。究其原因,发现是mapboxGL在绘图…...
AI重塑了我的工作流
阅读内容 Inhai: Agentic Workflow:AI 重塑了我的工作流 4 种主要的 Agentic Workflow 设计模式 Reflection(反思):让 Agent 审视和修正自己生成的输出。 举例:如果有两个 Agent:一个负责 Coding&#…...
vue使用Less报错semi-colon expectedcss(css-semicolonexpected)的解决方法
1、将 styleint 依赖项添加到项目中 npm install --save-dev stylelint stylelint-config-standard2、在根目录中添加stylelint.config.js文件(与package.json同级) module.exports {extends: ["stylelint-config-standard"],rules: {"…...
如何使用golang自带工具对代码进行覆盖率测试
在 Go 语言中,测试代码覆盖率通常使用 go test 命令结合 -cover 和 -coverprofile 1. 基本代码覆盖率报告 在项目目录下运行以下命令 go test -cover这将在控制台输出一个代码覆盖率的百分比。但是,这种方式不会保存覆盖率数据(可以指定目…...
Android studio版本和Android gradle plugin版本对应表
1.Android studio 版本的升级,一个方面上看主要是升级对AGP最高版本的支持 2.那为什么AGP要出高版本呢,主要支持高版本的API,真是一环扣一环...
质量体系的重要
质量体系是为确保产品、服务或过程质量满足规定要求,由相互关联的要素构成的有机整体。其核心内容可归纳为以下五个方面: 🏛️ 一、组织架构与职责 质量体系明确组织内各部门、岗位的职责与权限,形成层级清晰的管理网络…...
1.3 VSCode安装与环境配置
进入网址Visual Studio Code - Code Editing. Redefined下载.deb文件,然后打开终端,进入下载文件夹,键入命令 sudo dpkg -i code_1.100.3-1748872405_amd64.deb 在终端键入命令code即启动vscode 需要安装插件列表 1.Chinese简化 2.ros …...
2025 后端自学UNIAPP【项目实战:旅游项目】6、我的收藏页面
代码框架视图 1、先添加一个获取收藏景点的列表请求 【在文件my_api.js文件中添加】 // 引入公共的请求封装 import http from ./my_http.js// 登录接口(适配服务端返回 Token) export const login async (code, avatar) > {const res await http…...
【配置 YOLOX 用于按目录分类的图片数据集】
现在的图标点选越来越多,如何一步解决,采用 YOLOX 目标检测模式则可以轻松解决 要在 YOLOX 中使用按目录分类的图片数据集(每个目录代表一个类别,目录下是该类别的所有图片),你需要进行以下配置步骤&#x…...
CMake 从 GitHub 下载第三方库并使用
有时我们希望直接使用 GitHub 上的开源库,而不想手动下载、编译和安装。 可以利用 CMake 提供的 FetchContent 模块来实现自动下载、构建和链接第三方库。 FetchContent 命令官方文档✅ 示例代码 我们将以 fmt 这个流行的格式化库为例,演示如何: 使用 FetchContent 从 GitH…...
人机融合智能 | “人智交互”跨学科新领域
本文系统地提出基于“以人为中心AI(HCAI)”理念的人-人工智能交互(人智交互)这一跨学科新领域及框架,定义人智交互领域的理念、基本理论和关键问题、方法、开发流程和参与团队等,阐述提出人智交互新领域的意义。然后,提出人智交互研究的三种新范式取向以及它们的意义。最后,总结…...
七、数据库的完整性
七、数据库的完整性 主要内容 7.1 数据库的完整性概述 7.2 实体完整性 7.3 参照完整性 7.4 用户定义的完整性 7.5 触发器 7.6 SQL Server中数据库完整性的实现 7.7 小结 7.1 数据库的完整性概述 数据库完整性的含义 正确性 指数据的合法性 有效性 指数据是否属于所定…...
Web中间件--tomcat学习
Web中间件–tomcat Java虚拟机详解 什么是JAVA虚拟机 Java虚拟机是一个抽象的计算机,它可以执行Java字节码。Java虚拟机是Java平台的一部分,Java平台由Java语言、Java API和Java虚拟机组成。Java虚拟机的主要作用是将Java字节码转换为机器代码&#x…...
根目录0xa0属性对应的Ntfs!_SCB中的FileObject是什么时候被建立的----NTFS源代码分析--重要
根目录0xa0属性对应的Ntfs!_SCB中的FileObject是什么时候被建立的 第一部分: 0: kd> g Breakpoint 9 hit Ntfs!ReadIndexBuffer: f7173886 55 push ebp 0: kd> kc # 00 Ntfs!ReadIndexBuffer 01 Ntfs!FindFirstIndexEntry 02 Ntfs!NtfsUpda…...
comfyui 工作流中 图生视频 如何增加视频的长度到5秒
comfyUI 工作流怎么可以生成更长的视频。除了硬件显存要求之外还有别的方法吗? 在ComfyUI中实现图生视频并延长到5秒,需要结合多个扩展和技巧。以下是完整解决方案: 核心工作流配置(24fps下5秒120帧) #mermaid-svg-yP…...
