Swift:集合类型
Swift 语言提供 Arrays、Sets 和 Dictionaries 三种基本的集合类型用来存储集合数据。
数组(Arrays)是有序数据的集。
集合(Sets)是无序无重复数据的集。
字典(Dictionaries)是无序的键值对的集。
Swift 语言中的 Arrays、Sets 和 Dictionaries 中存储的数据值类型必须明确。
这意味着我们不能把错误的数据类型插入其中。
同时这也说明你完全可以对取回值的类型非常放心。
一、集合的可变性
-
如果创建一个
Arrays
、Sets
或Dictionaries
并且把它分配成一个变量,这个集合将会是可变的。 -
这意味着你可以在创建之后添加更多或移除已存在的数据项,或者改变集合中的数据项。
-
如果我们把
Arrays
、Sets
或Dictionaries
分配成常量,那么它就是不可变的,它的大小和内容都不能被改变。
二、数组
- 数组使用有序列表存储同一类型的多个值。相同的值可以多次出现在一个数组的不同位置中。
1、创建一个空数组
- 我们可以使用构造语法来创建一个由特定数据类型构成的空数组:
let arr0 = [Int]()
let arr1: [Int] = []
let arr2 = [] as [Int]
let arr3 = Array<Int>()
- 如果代码上下文中已经提供了类型信息,例如一个函数参数或者一个已经定义好类型的常量或者变量,我们可以使用空数组语句创建一个空数组,它的写法很简单:
[]
(一对空方括号)
arr.append(3)
// arr 现在包含一个 Int 值
arr = []
// arr 现在是空数组,但是仍然是 [Int] 类型的。
2、创建一个带有默认值的数组
- Swift 中的 Array 类型还提供一个可以创建特定大小并且所有数据都被默认的构造方法
var threeDoubles = Array(repeating: 0.0, count: 3)
// threeDoubles 是一种 [Double] 数组,等价于 [0.0, 0.0, 0.0]
3、通过两个数组相加创建一个数组
- 我们可以使用加法操作符(
+
)来组合两种已存在的相同类型数组。新数组的数据类型会被从两个数组的数据类型中推断出来:
var anotherThreeDoubles = Array(repeating: 2.5, count: 3)
// anotherThreeDoubles 被推断为 [Double],等价于 [2.5, 2.5, 2.5]
var sixDoubles = threeDoubles + anotherThreeDoubles
// sixDoubles 被推断为 [Double],等价于 [0.0, 0.0, 0.0, 2.5, 2.5, 2.5]
4、用数组字面量构造数组
- 我们可以使用数组字面量来进行数组构造,这是一种用一个或者多个数值构造数组的简单方法。数组字面量是一系列由逗号分割并由方括号包含的数值:
// [value 1, value 2, value 3]
var shoppingList: [String] = ["Eggs", "Milk"]
// shoppingList 已经被构造并且拥有两个初始项。
- 由于 Swift 的
类型推断机制
,当我们用字面量构造只拥有相同类型值
数组的时候,我们不必把数组的类型定义清楚
var shoppingList = ["Eggs", "Milk"]
5、访问和修改数组
-
我们可以通过数组的方法和属性来访问和修改数组,或者使用下标语法。
-
可以使用数组的只读属性
count
来获取数组中的数据项数量:
print("The shopping list contains \(shoppingList.count) items.")
// 输出 "The shopping list contains 2 items."(这个数组有2个项)
- 使用布尔属性
isEmpty
作为一个缩写形式去检查count
属性是否为 0:
if shoppingList.isEmpty {
print("The shopping list is empty.")
} else {
print("The shopping list is not empty.")
}
// 打印 "The shopping list is not empty."(shoppinglist 不是空的)
- 也可以使用
append(_:)
方法在数组后面添加新的数据项:
shoppingList.append("Flour")
// shoppingList 现在有3个数据项,有人在摊煎饼
- 除此之外,使用加法赋值运算符(
+=
)也可以直接在数组后面添加一个或多个拥有相同类型的数据项:
shoppingList += ["Baking Powder"]
// shoppingList 现在有四项了
shoppingList += ["Chocolate Spread", "Cheese", "Butter"]
// shoppingList 现在有七项了
- 可以直接使用下标语法来获取数组中的数据项,把我们需要的数据项的索引值放在直接放在数组名称的方括号中:
var firstItem = shoppingList[0]
// 第一项是 "Eggs"
- 我们也可以用下标来改变某个已有索引值对应的数据值:
shoppingList[0] = "Six eggs"
// 其中的第一项现在是 "Six eggs" 而不是 "Eggs"
- 还可以利用下标来一次改变一系列数据值,即使新数据和原有数据的数量是不一样的。下面的例子把 "Chocolate Spread"、"Cheese" 和 "Butter" 替换为 "Bananas" 和 "Apples":
shoppingList[4...6] = ["Bananas", "Apples"]
// shoppingList 现在有6项
注意
不可以用下标访问的形式去在数组尾部添加新项。
- 下面的代码是错误的
var a = [Int]()
for i in 0..<10 {
a[i] = i // 报错" Thread 1: Fatal error: Index out of range
}
print(a)
- 调用数组的
insert(_:at:)
方法来在某个具体索引值之前添加数据项:
shoppingList.insert("Maple Syrup", at: 0)
// shoppingList 现在有7项
// "Maple Syrup" 现在是这个列表中的第一项
- 我们可以使用
insert(_:at:)
逐个索引插入数据
var a = [Int]()
for i in 0..<10 {
a.insert(i, at: i)
}
print(a) // 打印: [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
- 类似的我们可以使用
remove(at:)
方法来移除数组中的某一项。这个方法把数组在特定索引值中存储的数据项移除并且返回这个被移除的数据项(我们不需要的时候就可以无视它):
let mapleSyrup = shoppingList.remove(at: 0)
// 索引值为0的数据项被移除
// shoppingList 现在只有6项,而且不包括 Maple Syrup
// mapleSyrup 常量的值等于被移除数据项的值 "Maple Syrup"
- 数据项被移除后数组中的空出项会被自动填补,所以现在索引值为 0 的数据项的值再次等于 "Six eggs":
firstItem = shoppingList[0]
// firstItem 现在等于 "Six eggs"
- 如果我们只想把数组中的最后一项移除,可以使用
removeLast()
方法, 并返回移除的数据项
let apples = shoppingList.removeLast()
// 数组的最后一项被移除了
// shoppingList 现在只有5项,不包括 Apples
// apples 常量的值现在等于 "Apples" 字符串
6、数组的遍历
- 我们可以使用 for-in 循环来遍历所有数组中的数据项:
for item in shoppingList {
print(item)
}
// Six eggs
// Milk
// Flour
// Baking Powder
// Bananas
- 如果我们同时需要每个数据项的值和索引值,可以使用
enumerated()
方法来进行数组遍历。
for (index, value) in shoppingList. enumerated() {
print("Item \(String(index + 1)): \(value)")
}
// Item 1: Six eggs
// Item 2: Milk
// Item 3: Flour
// Item 4: Baking Powder
// Item 5: Bananas
三、集合
- 集合(Set)用来存储相同类型并且没有确定顺序的值。当集合元素顺序不重要时或者希望确保每个元素只出现一次时可以使用集合而不是数组。
1、集合类型的哈希值
- 一个类型为了存储在集合中,该类型必须是可哈希化的
- Swift 的所有基本类型(比如
String
,Int
,Double
和Bool
)默认都是可哈希化的,可以作为集合的值的类型或者字典的键的类型。没有关联值的枚举成员值, 默认也是可哈希化的。
注意
你可以使用你自定义的类型作为集合的值的类型或者是字典的键的类型,但你需要使你的自定义类型符合 Swift 标准库中的 Hashable 协议。
2、集合类型语法
- Swift 中的
Set
类型被写为Set<Element>
,这里的Element
表示Set
中允许存储的类型,和数组不同的是,集合没有等价的简化形式。
3、创建和构造一个空的集合
- 你可以通过构造器语法创建一个特定类型的空集合:
var letters = Set<Character>()
print("letters is of type Set<Character> with \(letters.count) items.")
// 打印 "letters is of type Set<Character> with 0 items."
- 此外,如果上下文提供了类型信息,比如作为函数的参数或者已知类型的变量或常量,我们可以通过一个空的数组字面量创建一个空的 Set:
letters.insert("a")
// letters 现在含有1个 Character 类型的值
letters = []
// letters 现在是一个空的 Set, 但是它依然是 Set<Character> 类型
4、用数组字面量创建集合
- 你可以使用数组字面量来构造集合,并且可以使用简化形式写一个或者多个值作为集合元素。
var favoriteGenres: Set<String> = ["Rock", "Classical", "Hip hop"]
// favoriteGenres 被构造成含有三个初始值的集合
-
一个
Set
类型不能从数组字面量中被单独推断出来,因此Set
类型必须显式声明 -
然而,由于 Swift 的类型推断功能,如果你想使用一个数组字面量构造一个
Set
并且该数组字面量中的所有元素类型相同,那么你无须写出Set
的具体类型。
var favoriteGenres: Set = ["Rock", "Classical", "Hip hop"]
5、访问和修改一个集合
- 你可以通过
Set
的属性和方法来访问和修改一个Set
。 - 为了找出一个
Set
中元素的数量,可以使用其只读属性count
:
print("I have \(favoriteGenres.count) favorite music genres.")
// 打印 "I have 3 favorite music genres."
- 用布尔属性
isEmpty
作为一个缩写形式去检查count
属性是否为0
:
if favoriteGenres.isEmpty {
print("As far as music goes, I'm not picky.")
} else {
print("I have particular music preferences.")
}
// 打印 "I have particular music preferences."
- 可以通过调用
Set
的insert(_:)
方法来添加一个新元素:
favoriteGenres.insert("Jazz")
// favoriteGenres 现在包含4个元素
- 可以通过调用
Set
的remove(_:)
方法去删除一个元素,如果该值是该Set
的一个元素则删除该元素并且返回被删除的元素值,否则如果该Set
不包含该值,则返回nil
。另外,Set
中的所有元素可以通过它的removeAll()
方法删除。
if let removedGenre = favoriteGenres.remove("Rock") {
print("\(removedGenre)? I'm over it.")
} else {
print("I never much cared for that.")
}
// 打印 "Rock? I'm over it."
- 使用
contains(_:)
方法去检查Set
中是否包含一个特定的值:
if favoriteGenres.contains("Funk") {
print("I get up on the good foot.")
} else {
print("It's too funky in here.")
}
// 打印 "It's too funky in here."
6、遍历一个集合
- 你可以在一个 for-in 循环中遍历一个 Set 中的所有值。
for genre in favoriteGenres {
print("\(genre)")
}
// Classical
// Jazz
// Hip hop
- Swift 的
Set
类型没有确定的顺序,为了按照特定顺序来遍历一个Set
中的值可以使用sorted()
方法,它将返回一个有序数组,这个数组的元素排列顺序由操作符'<'对元素进行比较的结果来确定。
for genre in favoriteGenres.sorted() {
print("\(genre)")
}
// prints "Classical"
// prints "Hip hop"
// prints "Jazz
四、集合操作
- 你可以高效地完成
Set
的一些基本操作,比如把两个集合组合到一起,判断两个集合共有元素,或者判断两个集合是否全包含,部分包含或者不相交。
1、基本集合操作
- 交集:
intersection(_:)
- 并集:
union(_:)
- 对称差集:
symmetricDifference(_:)
- 差集:
subtracting(_:)
let a: Set = [1, 3, 5, 7, 9]
let b: Set = [0, 2, 4, 6, 8]
let c: Set = [2, 3, 5, 7]
// 交集 同时即在a中, 也在c中
result = a.intersection(c).sorted()
print(result) // 打印: [3, 5, 7]
// 并集 a中和b中
var result = a.union(b).sorted()
print(result) // 打印: [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
// 对称差集 在a中或在c中, 但不同时在a和c中
result = a.symmetricDifference(c).sorted()
print(result) // 打印: [1, 2, 9]
// 差集 在a中而不在c中
result = a.subtracting(c).sorted()
print(result) // 打印: [1, 9]
2、集合成员关系和相等
- 使用“是否相等”运算符(
==
)来判断两个集合是否包含全部相同的值。 - 使用
isSubset(of:)
方法来判断一个集合中的值是否也被包含在另外一个集合中。 - 使用
isSuperset(of:)
方法来判断一个集合中包含另一个集合中所有的值。 - 使用
isStrictSubset(of:)
或者isStrictSuperset(of:)
方法来判断一个集合是否是另外一个集合的子集合或者父集合并且两个集合并不相等。 - 使用
isDisjoint(with:)
方法来判断两个集合是否不含有相同的值(是否没有交集)。
let houseAnimals: Set = ["🐶", "🐱"]
let farmAnimals: Set = ["🐮", "🐔", "🐑", "🐶", "🐱"]
let cityAnimals: Set = ["🐦", "🐭"]
houseAnimals.isSubset(of: farmAnimals)
// true
farmAnimals.isSuperset(of: houseAnimals)
// true
farmAnimals.isDisjoint(with: cityAnimals)
// true
五、字典
-
字典是一种存储多个相同类型的值的容器。每个值(value)都关联唯一的键(key),键作为字典中的这个值数据的标识符。
-
和数组中的数据项不同,字典中的数据项并没有具体顺序。我们在需要通过标识符(键)访问数据的时候使用字典
1、字典类型简化语法
- Swift 的字典使用
Dictionary<Key, Value>
定义,其中Key
是字典中键的数据类型,Value
是字典中对应于这些键所存储值的数据类型。 - 我们也可以用
[Key: Value]
这样简化的形式去创建一个字典类型。
2、创建一个空字典
- 以
[Int: String]
类型为例
let dict1 = Dictionary<Int, String>()
let dict2 = [Int:String]()
let dict3: [Int:String] = [:]
let dict4 = [:] as [Int:String]
3、用字典字面量创建字典
- 语法
[key 1: value 1, key 2: value 2, key 3: value 3]
- 下面的例子创建了一个存储国际机场名称的字典。
var airports: [String: String] = ["YYZ": "Toronto Pearson", "DUB": "Dublin"]
- 和数组一样,我们在用字典字面量构造字典时,如果它的键和值都有各自一致的类型,那么就不必写出字典的类型。
var airports = ["YYZ": "Toronto Pearson", "DUB": "Dublin"]
4、访问和修改字典
- 我们可以通过字典的只读属性
count
来获取某个字典的数据项数量:
print("The dictionary of airports contains \(airports.count) items.")
// 打印 "The dictionary of airports contains 2 items."(这个字典有两个数据项)
- 使用布尔属性
isEmpty
作为一个缩写形式去检查count
属性是否为0
:
if airports.isEmpty {
print("The airports dictionary is empty.")
} else {
print("The airports dictionary is not empty.")
}
// 打印 "The airports dictionary is not empty."
- 我们也可以在字典中使用下标语法来添加新的数据项。
airports["LHR"] = "London"
// airports 字典现在有三个数据项
- 我们也可以使用下标语法来改变特定键对应的值:
airports["LHR"] = "London Heathrow"
// “LHR”对应的值被改为“London Heathrow”
-
作为另一种下标方法,字典的
updateValue(_:forKey:)
方法可以设置或者更新特定键对应的值。 -
updateValue(_:forKey:)
方法在这个键不存在对应值的时候会设置新值或者在存在时更新已存在的值。 -
updateValue(_:forKey:)
这个方法返回更新值之前的原值。这样使得我们可以检查更新是否成功。 -
updateValue(_:forKey:)
方法会返回对应值的类型的可选值。 -
如果有值存在于更新前,则这个可选值包含了旧值,否则它将会是
nil
。
if let oldValue = airports.updateValue("Dublin Airport", forKey: "DUB") {
print("The old value for DUB was \(oldValue).")
}
// 输出 "The old value for DUB was Dublin."
-
我们也可以使用下标语法来在字典中检索特定键对应的值。因为有可能请求的键没有对应的值存在,字典的下标访问会返回对应值的类型的可选值。
-
如果这个字典包含请求键所对应的值,下标会返回一个包含这个存在值的可选值,否则将返回
nil
:
if let airportName = airports["DUB"] {
print("The name of the airport is \(airportName).")
} else {
print("That airport is not in the airports dictionary.")
}
// 打印 "The name of the airport is Dublin Airport."
- 我们还可以使用下标语法来通过给某个键的对应值赋值为
nil
来从字典里移除一个键值对:
airports["APL"] = "Apple Internation"
// "Apple Internation" 不是真的 APL 机场,删除它
airports["APL"] = nil
// APL 现在被移除了
- 此外,
removeValue(forKey:)
方法也可以用来在字典中移除键值对。这个方法在键值对存在的情况下会移除该键值对并且返回被移除的值或者在没有值的情况下返回nil
:
if let removedValue = airports. removeValue(forKey: "DUB") {
print("The removed airport's name is \(removedValue).")
} else {
print("The airports dictionary does not contain a value for DUB.")
}
// prints "The removed airport's name is Dublin Airport."
5、字典遍历
- 我们可以使用
for-in
循环来遍历某个字典中的键值对。每一个字典中的数据项都以(key, value)
元组形式返回,并且我们可以使用临时常量或者变量来分解这些元组:
for (airportCode, airportName) in airports {
print("\(airportCode): \(airportName)")
}
// YYZ: Toronto Pearson
// LHR: London Heathrow
- 通过访问
keys
或者values
属性,我们也可以遍历字典的键或者值:
for airportCode in airports.keys {
print("Airport code: \(airportCode)")
}
// Airport code: YYZ
// Airport code: LHR
for airportName in airports.values {
print("Airport name: \(airportName)")
}
// Airport name: Toronto Pearson
// Airport name: London Heathrow
- Swift 的字典类型是无序集合类型。为了以特定的顺序遍历字典的键或值,可以对字典的
keys
或values
属性使用sorted()
方法。