> 1.0 翻譯:[zqp](https://github.com/zqp)?校對:[shinyzhu](https://github.com/shinyzhu),?[stanzhai](https://github.com/stanzhai),?[feiin](https://github.com/feiin)
>
> 2.0 翻譯+校對:[JackAlan](https://github.com/AlanMelody)
* * *
本頁包含內容:
[TOC=2]
Swift 語言提供`Arrays`、`Sets`和`Dictionaries`三種基本的集合類型用來存儲集合數據。數組是有序數據的集。集合是無序無重復數據的集。字典是無序的鍵值對的集。

Swift 語言中的`Arrays`、`Sets`和`Dictionaries`中存儲的數據值類型必須明確。這意味著我們不能把不正確的數據類型插入其中。同時這也說明我們完全可以對取回值的類型非常自信。
> 注意: Swift 的`Arrays`、`Sets`和`Dictionaries`類型被實現為泛型集合。更多關于泛型類型和集合,參見?[泛型](http://wiki.jikexueyuan.com/project/swift/chapter2/23_Generics.html)章節。
## 集合的可變性
如果創建一個`Arrays`、`Sets`或`Dictionaries`并且把它分配成一個變量,這個集合將會是可變的。這意味著我們可以在創建之后添加更多或移除已存在的數據項來改變這個集合的大小。如果我們把`Arrays`、`Sets`或`Dictionaries`分配成常量,那么它就是不可變的,它的大小不能被改變。
> 注意: 在我們不需要改變集合大小的時候創建不可變集合是很好的習慣。如此 Swift 編譯器可以優化我們創建的集合。
## 數組(Arrays)
數組使用有序列表存儲同一類型的多個值。相同的值可以多次出現在一個數組的不同位置中。
> 注意: Swift 的`Array`類型被橋接到`Foundation`中的`NSArray`類。 更多關于在`Foundation`和`Cocoa`中使用`Array`的信息,參見?[_Using Swift with Cocoa and Obejective-C_](https://developer.apple.com/library/prerelease/ios/documentation/Swift/Conceptual/BuildingCocoaApps/index.html#//apple_ref/doc/uid/TP40014216)?一書。
### 數組的簡單語法
寫 Swift 數組應該遵循像`Array<T>`這樣的形式,其中`T`是這個數組中唯一允許存在的數據類型。我們也可以使用像`[T]`這樣的簡單語法。盡管兩種形式在功能上是一樣的,但是推薦較短的那種,而且在本文中都會使用這種形式來使用數組。
### 創建一個空數組
我們可以使用構造語法來創建一個由特定數據類型構成的空數組:
~~~
var someInts = [Int]()
print("someInts is of type [Int] with \(someInts.count) items。")
// 打印 "someInts is of type [Int] with 0 items。"
~~~
注意 通過構造函數的類型,`someInts`的值類型被推斷為`[Int]`。
或者,如果代碼上下文中已經提供了類型信息,例如一個函數參數或者一個已經定義好類型的常量或者變量,我們可以使用空數組語句創建一個空數組,它的寫法很簡單:`[]`(一對空方括號):
~~~
someInts.append(3)
// someInts 現在包含一個Int值
someInts = []
// someInts 現在是空數組,但是仍然是[Int]類型的。
~~~
### 創建一個帶有默認值的數組
Swift 中的`Array`類型還提供一個可以創建特定大小并且所有數據都被默認的構造方法。我們可以把準備加入新數組的數據項數量(`count`)和適當類型的初始值(`repeatedValue`)傳入數組構造函數:
~~~
var threeDoubles = [Double](count: 3, repeatedValue:0.0)
// threeDoubles 是一種 [Double]數組, 等于 [0.0, 0.0, 0.0]
~~~
### 通過兩個數組相加創建一個數組
我們可以使用加法操作符(`+`)來組合兩種已存在的相同類型數組。新數組的數據類型會被從兩個數組的數據類型中推斷出來:
~~~
var anotherThreeDoubles = Array(count: 3, repeatedValue: 2.5)
// anotherThreeDoubles is inferred as [Double], and equals [2.5, 2.5, 2.5]
var sixDoubles = threeDoubles + anotherThreeDoubles
// sixDoubles 被推斷為 [Double], 等于 [0.0, 0.0, 0.0, 2.5, 2.5, 2.5]
~~~
### 用字面量構造數組
我們可以使用字面量來進行數組構造,這是一種用一個或者多個數值構造數組的簡單方法。字面量是一系列由逗號分割并由方括號包含的數值。?`[value 1, value 2, value 3]`。
下面這個例子創建了一個叫做`shoppingList`并且存儲`String`的數組:
~~~
var shoppingList: [String] = ["Eggs", "Milk"]
// shoppingList 已經被構造并且擁有兩個初始項。
~~~
`shoppingList`變量被聲明為“字符串值類型的數組“,記作`[String]`。 因為這個數組被規定只有`String`一種數據結構,所以只有`String`類型可以在其中被存取。 在這里,`shoppinglist`數組由兩個`String`值(`"Eggs"`?和`"Milk"`)構造,并且由字面量定義。
> 注意:?`Shoppinglist`數組被聲明為變量(`var`關鍵字創建)而不是常量(`let`創建)是因為以后可能會有更多的數據項被插入其中。
在這個例子中,字面量僅僅包含兩個`String`值。匹配了該數組的變量聲明(只能包含`String`的數組),所以這個字面量的分配過程就是允許用兩個初始項來構造`shoppinglist`。
由于 Swift 的類型推斷機制,當我們用字面量構造只擁有相同類型值數組的時候,我們不必把數組的類型定義清楚。`shoppinglist`的構造也可以這樣寫:
~~~
var shoppingList = ["Eggs", "Milk"]
~~~
因為所有字面量中的值都是相同的類型,Swift 可以推斷出`[String]`是`shoppinglist`中變量的正確類型。
### 訪問和修改數組
我們可以通過數組的方法和屬性來訪問和修改數組,或者下標語法。 還可以使用數組的只讀屬性`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"
~~~
注意第一項在數組中的索引值是`0`而不是`1`。 Swift 中的數組索引總是從零開始。
我們也可以用下標來改變某個已有索引值對應的數據值:
~~~
shoppingList[0] = "Six eggs"
// 其中的第一項現在是 "Six eggs" 而不是 "Eggs"
~~~
還可以利用下標來一次改變一系列數據值,即使新數據和原有數據的數量是不一樣的。下面的例子把`"Chocolate Spread"`,`"Cheese"`,和`"Butter"`替換為`"Bananas"`和?`"Apples"`:
~~~
shoppingList[4...6] = ["Bananas", "Apples"]
// shoppingList 現在有六項
~~~
> 注意: 不可以用下表訪問的形式去在數組尾部添加新項。
調用數組的`insert(_:atIndex:)`方法來在某個具體索引值之前添加數據項:
~~~
shoppingList.insert("Maple Syrup", atIndex: 0)
// shoppingList 現在有7項
// "Maple Syrup" 現在是這個列表中的第一項
~~~
這次`insert`函數調用把值為`"Maple Syrup"`的新數據項插入列表的最開始位置,并且使用`0`作為索引值。
類似的我們可以使用`removeAtIndex`方法來移除數組中的某一項。這個方法把數組在特定索引值中存儲的數據項移除并且返回這個被移除的數據項(我們不需要的時候就可以無視它):
~~~
let mapleSyrup = shoppingList.removeAtIndex(0)
// 索引值為0的數據項被移除
// shoppingList 現在只有6項,而且不包括Maple Syrup
// mapleSyrup常量的值等于被移除數據項的值 "Maple Syrup"
~~~
> 注意: 如果我們試著對索引越界的數據進行檢索或者設置新值的操作,會引發一個運行期錯誤。我們可以使用索引值和數組的`count`屬性進行比較來在使用某個索引之前先檢驗是否有效。除了當`count`等于 0 時(說明這是個空數組),最大索引值一直是`count - 1`,因為數組都是零起索引。
數據項被移除后數組中的空出項會被自動填補,所以現在索引值為`0`的數據項的值再次等于`"Six eggs"`:
~~~
firstItem = shoppingList[0]
// firstItem 現在等于 "Six eggs"
~~~
如果我們只想把數組中的最后一項移除,可以使用`removeLast`方法而不是`removeAtIndex(_:)`方法來避免我們需要獲取數組的`count`屬性。就像后者一樣,前者也會返回被移除的數據項:
~~~
let apples = shoppingList.removeLast()
// 數組的最后一項被移除了
// shoppingList現在只有5項,不包括cheese
// apples 常量的值現在等于"Apples" 字符串
~~~
### 數組的遍歷
我們可以使用`for-in`循環來遍歷所有數組中的數據項:
~~~
for item in shoppingList {
print(item)
}
// Six eggs
// Milk
// Flour
// Baking Powder
// Bananas
~~~
如果我們同時需要每個數據項的值和索引值,可以使用`enumerate()`方法來進行數組遍歷。`enumerate()`返回一個由每一個數據項索引值和數據值組成的元組。我們可以把這個元組分解成臨時常量或者變量來進行遍歷:
~~~
for (index, value) in shoppingList.enumerate() {
print("Item \(String(index + 1)): \(value)")
}
// Item 1: Six eggs
// Item 2: Milk
// Item 3: Flour
// Item 4: Baking Powder
// Item 5: Bananas
~~~
更多關于`for-in`循環的介紹請參見[for 循環](http://wiki.jikexueyuan.com/project/swift/chapter2/05_Control_Flow.html#for_loops)。
## 集合
集合(Set)用來存儲相同類型并且沒有確定順序的值。當集合元素順序不重要時或者希望確保每個元素只出現一次時可以把集合當做是數組另一形式。
> 注意: Swift的`Set`類型被橋接到`Fundation`中的`NSSet`類。 關于使用`Fundation`和`Cocoa`中`Set`的知識,請看[_Using Swift with Cocoa and Objective-C_](https://developer.apple.com/library/prerelease/ios/documentation/Swift/Conceptual/BuildingCocoaApps/index.html#//apple_ref/doc/uid/TP40014216)。
#### Set類型的哈希值
為了存儲在集合中,該類型必須是可哈希化的-也就是說,該類型必須提供一個方法來計算它的哈希值。一個哈希值是`Int`類型的,它和其他的對象相同,其被用來比較相等與否,比如`a==b`,它遵循的是`a.hashValue == b.hashValue`。
Swift 的所有基本類型(比如`String`,`Int`,`Double`和`Bool`)默認都是可哈希化的,它可以作為集合的值或者字典的鍵值類型。沒有關聯值的枚舉成員值(在[枚舉](http://wiki.jikexueyuan.com/project/swift/chapter2/08_Enumerations.html)有講述)默認也是可哈希化的。
> 注意: 你可以使用你自定義的類型作為集合的值或者是字典的鍵值類型,但你需要使你的自定義類型服從Swift標準庫中的`Hashable`協議。服從`Hashable`協議的類型需要提供一個類型為`Int`的取值訪問器屬性`hashValue`。這個由類型的`hashValue`返回的值不需要在同一程序的不同執行周期或者不同程序之間保持相同。 因為`hashable`協議服從于`Equatable`協議,所以遵循該協議的類型也必須提供一個"是否等"運算符(`==`)的實現。這個`Equatable`協議需要任何遵循的`==`的實現都是一種相等的關系。也就是說,對于`a,b,c`三個值來說,`==`的實現必須滿足下面三種情況:
* `a==a`(自反性)
* `a==b`意味著`b==a`(對稱性)
* `a==b&&b==c`意味著`a==c`(傳遞性)
關于協議遵循的更多信息,請看[協議](http://wiki.jikexueyuan.com/project/swift/chapter2/22_Protocols.html)
### Set類型語法
Swift中的`Set`類型被寫為`Set<T>`, 這里的`T`表示`Set`中允許存儲的類型,和數組不同的是,集合沒有等價的簡化形式。
### 創建和構造一個空的Set
你可以通過構造器語法創建一個特定類型的空集合:
~~~
var letters = Set<Character>()
print("letters is of type Set<Character> with \(letters.count) items.")
// 打印 "letters is of type Set<Character> with 0 items."
~~~
> 注意: 通過構造器,這里的`letters`變量的類型被推斷為`Set<Character>`。
此外,如果上下文提供了類型信息,比如作為函數的參數或者已知類型的變量或常量,我們可以通過一個空的數組字面量創建一個空的`Set`:
~~~
letters.insert("a")
// letters 現在含有1個Character類型的值
letters = []
// letters 現在是一個空的Set, 但是它依然是 Set<Character> 類型
~~~
### 用數組字面量創建集合
你可以使用數組字面量來構造集合,并且可以使用簡化形式寫一個或者多個值作為集合元素。
下面的例子創建一個稱之為`favoriteGenres`的集合來存儲`String`類型的值:
~~~
var favoriteGenres: Set<String> = ["Rock", "Classical", "Hip hop"]
// favoriteGenres被構造成含有三個初始值的集合
~~~
這個`favoriteGenres`變量被聲明為“一個`String`值的集合”,寫為`Set<String>`。由于這個特定的集合含有指定`String`類型的值,所以它只允許存儲`String`類型值。這里的`favoriteGenres`變量有三個`String`類型的初始值("`Rock`","`Classical`"和"`Hip hop`"),并以數組字面量的方式出現。
> 注意:?`favoriteGenres`被聲明為一個變量(擁有`var`標示符)而不是一個常量(擁有`let`標示符),因為它里面的元素將會在下面的例子中被增加或者移除。
一個`Set`類型不能從數組中字面量中獨立地被推斷出來,因此`Set`類型必須顯式聲明。然而,由于Swift的類型推導功能,如果你想使用一個數組字面量構造一個Set并且該數組字面量中的所有元素類型相同,那么你無須寫出`Set`的具體類型。`favoriteGenres`的構造形式可以采用簡化的方式代替:
~~~
var favoriteGenres: Set = ["Rock", "Classical", "Hip hop"]
~~~
由于數組字面量中的所有元素類型相同,Swift可以推斷出`Set<String>`作為`favoriteGenres`變量的正確類型。
### 訪問和修改一個Set
你可以通過`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."
~~~
### 遍歷一個Set
你可以在一個`for-in`循環中遍歷一個`Set`中的所有值。
~~~
for genre in favoriteGenres {
print("\(genre)")
}
// Classical
// Jazz
// Hip hop
~~~
更多關于`for-in`循環信息,參見[For循環](http://wiki.jikexueyuan.com/project/swift/chapter2/05_Control_Flow.html#for_loops)。
Swift 的`Set`類型沒有確定的順序,為了按照特定順序來遍歷一個`Set`中值可以使用`sort()`方法,它將根據提供的序列返回一個排序的集合.
~~~
for genre in favoriteGenres.sort() {
print("\(genre)")
}
// prints "Classical"
// prints "Hip hop"
// prints "Jazz
~~~
### 完成集合操作
你可以高效的完成`Set`的一些基本操作,比如把兩個集合組合到一起,判斷兩個集合共有元素,或者判斷兩個集合是否全包含,部分包含或者不相交。
#### 基本集合操作
下面的插圖描述了兩個集合-`a`和`b`-以及通過陰影部分的區域顯示集合各種操作的結果。

* 使用`intersect(_:)`方法根據兩個集合中都包含的值創建的一個新的集合。
* 使用`exclusiveOr(_:)`方法根據值在一個集合中但不在兩個集合中的值創建一個新的集合。
* 使用`union(_:)`方法根據兩個集合的值創建一個新的集合。
* 使用`subtract(_:)`方法根據不在該集合中的值創建一個新的集合。
~~~
let oddDigits: Set = [1, 3, 5, 7, 9]
let evenDigits: Set = [0, 2, 4, 6, 8]
let singleDigitPrimeNumbers: Set = [2, 3, 5, 7]
oddDigits.union(evenDigits).sort()
// [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
oddDigits.intersect(evenDigits).sort()
// []
oddDigits.subtract(singleDigitPrimeNumbers).sort()
// [1, 9]
oddDigits.exclusiveOr(singleDigitPrimeNumbers).sort()
// [1, 2, 9]
~~~
#### 集合成員關系和相等
下面的插圖描述了三個集合-`a`,`b`和`c`,以及通過懸浮區域表述集合間共享的元素。Set?`a`是Set`b`的父集合,因為`a`包含了`b`中所有的元素,相反的,Set?`b`是`a`的子集合,因為屬于`b`的元素也被`a`包含。Set?`b`和Set?`c`彼此不關聯,因為它們之間沒有共同的元素。

* 使用“是否等”運算符(`==`)來判斷兩個集合是否包含全部相同的值。
* 使用`isSubsetOf(_:)`方法來判斷一個集合中的值是否也被包含在另外一個集合中。
* 使用`isSupersetOf(_:)`方法來判斷一個集合中包含的值是另一個集合中所有的值。
* 使用`isStrictSubsetOf(_:)`或者`isStrictSupersetOf(_:)`方法來判斷一個集合是否是另外一個集合的子集合或者父集合并且和特定集合不相等。
* 使用`isDisjointWith(_:)`方法來判斷兩個結合是否不含有相同的值。
~~~
let houseAnimals: Set = ["?", "?"]
let farmAnimals: Set = ["?", "?", "?", "?", "?"]
let cityAnimals: Set = ["?", "?"]
houseAnimals.isSubsetOf(farmAnimals)
// true
farmAnimals.isSupersetOf(houseAnimals)
// true
farmAnimals.isDisjointWith(cityAnimals)
// true
~~~
## 字典
字典是一種存儲多個相同類型的值的容器。每個值(value)都關聯唯一的鍵(key),鍵作為字典中的這個值數據的標識符。和數組中的數據項不同,字典中的數據項并沒有具體順序。我們在需要通過標識符(鍵)訪問數據的時候使用字典,這種方法很大程度上和我們在現實世界中使用字典查字義的方法一樣。
> 注意: Swiftly 的`Dictionary`?類型被橋接到Foundation的`NSDictionary`類。 更多關于在`Foundation`和`Cocoa`中使用`Dictionary`類型的信息,參見?[_Using Swift with Cocoa and Obejective-C_](https://developer.apple.com/library/prerelease/ios/documentation/Swift/Conceptual/BuildingCocoaApps/index.html#//apple_ref/doc/uid/TP40014216)?一書。
## 字典類型快捷語法
Swift 的字典使用`Dictionary<Key, Value>`定義,其中`Key`是字典中鍵的數據類型,`Value`是字典中對應于這些鍵所存儲值的數據類型。
> 注意: 一個字典的`Key`類型必須遵循`Hashable`協議,就像`Set`的值類型。
我們也可以用`[Key: Value]`這樣快捷的形式去創建一個字典類型。雖然這倆種形式功能上相同,但是后者是首選,并且這本指導書涉及到字典類型時通篇采用后者。
### 創建一個空字典
我們可以像數組一樣使用構造語法創建一個擁有確定類型的空字典:
~~~
var namesOfIntegers = [Int: String]()
// namesOfIntegers 是一個空的 [Int: String] 字典
~~~
這個例子創建了一個`[Int: String]`類型的空字典來儲存英語對整數的命名。它的鍵是`Int`型,值是`String`型。
如果上下文已經提供了信息類型,我們可以使用空字典字面量來創建一個空字典,記作`[:]`(中括號中放一個冒號):
~~~
namesOfIntegers[16] = "sixteen"
// namesOfIntegers 現在包含一個鍵值對
namesOfIntegers = [:]
// namesOfIntegers 又成為了一個 Int, String類型的空字典
~~~
## 字典字面量
我們可以使用字典字面量來構造字典,這和我們剛才介紹過的數組字面量擁有相似語法。字典字面量是一種將寫一個或多個鍵值對作`Dictionary`集合的快捷途徑。
一個鍵值對是一個`key`和一個`value`的結合體。在字典字面量中,每一個鍵值對的鍵和值都由冒號分割。這些鍵值對構成一個列表,其中這些鍵值對由方括號包含、由逗號分割:
~~~
[key 1: value 1, key 2: value 2, key 3: value 3]
~~~
下面的例子創建了一個存儲國際機場名稱的字典。在這個字典中鍵是三個字母的國際航空運輸相關代碼,值是機場名稱:
~~~
var airports: [String: String] = ["YYZ": "Toronto Pearson", "DUB": "Dublin"]
~~~
`airports`字典被聲明為一種?`[String: String]`類型,這意味著這個字典的鍵和值都是`String`類型。
> 注意:?`airports`字典被聲明為變量(用`var`關鍵字)而不是常量(`let`關鍵字)因為后來更多的機場信息會被添加到這個示例字典中。
`airports`字典使用字典字面量初始化,包含兩個鍵值對。第一對的鍵是`YYZ`,值是`Toronto Pearson`。第二對的鍵是`DUB`,值是`Dublin`。
這個字典語句包含了兩個`String: String`類型的鍵值對。它們對應`airports`變量聲明的類型(一個只有`String`鍵和`String`值的字典)所以這個字典字面量的任務是構造擁有兩個初始數據項的`airport`字典。
和數組一樣,我們在用字典字面量構造字典時,如果它的鍵和值都有各自一致的類型,那么就不必寫出字典的類型。`airports`的也可以用這種方法簡短定義:
~~~
var airports = ["YYZ": "Toronto Pearson", "DUB": "Dublin"]
~~~
因為這個語句中所有的鍵和值都各自擁有相同的數據類型,Swift 可以推斷出`Dictionary<String, String>`是`airports`字典的正確類型。
### 讀取和修改字典
我們可以通過字典的方法和屬性來讀取和修改字典,或者通過使用下標語法。
和數組一樣,我們可以通過字典的只讀屬性`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:)`函數會返回包含一個字典值類型的可選值。舉例來說:對于存儲`String`值的字典,這個函數會返回一個`String?`或者“可選?`String`”類型的值。
如果有值存在于更新前,則這個可選值包含了舊值,否則它將會是`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現在被移除了
~~~
此外,`removeValueForKey(_:)`方法也可以用來在字典中移除鍵值對。這個方法在鍵值對存在的情況下會移除該鍵值對并且返回被移除的值或者在沒有值的情況下返回`nil`:
~~~
if let removedValue = airports.removeValueForKey("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."
~~~
### 字典遍歷
我們可以使用`for-in`循環來遍歷某個字典中的鍵值對。每一個字典中的數據項都以`(key, value)`元組形式返回,并且我們可以使用臨時常量或者變量來分解這些元組:
~~~
for (airportCode, airportName) in airports {
print("\(airportCode): \(airportName)")
}
// YYZ: Toronto Pearson
// LHR: London Heathrow
~~~
更多關于`for-in`循環的信息,參見[For 循環](http://wiki.jikexueyuan.com/project/swift/chapter2/05_Control_Flow.html#for_loops)。
通過訪問`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
~~~
如果我們只是需要使用某個字典的鍵集合或者值集合來作為某個接受`Array`實例 API 的參數,可以直接使用`keys`或者`values`屬性直接構造一個新數組:
~~~
let airportCodes = Array(airports.keys)
// airportCodes is ["YYZ", "LHR"]
let airportNames = Array(airports.values)
// airportNames is ["Toronto Pearson", "London Heathrow"]
~~~
Swift 的字典類型是無序集合類型。為了以特定的順序遍歷字典的鍵或值,可以對字典的`keys`或`values`屬性使用`sort()`方法。
- 介紹
- 歡迎使用 Swift
- 關于 Swift
- Swift 初見
- Swift 版本歷史記錄
- Swift1.0 發布內容
- Swift 教程
- 基礎部分
- 基本運算符
- 字符串和字符
- 集合類型
- 控制流
- 函數
- 閉包
- 枚舉
- 類和結構體
- 屬性
- 方法
- 下標腳本
- 繼承
- 構造過程
- 析構過程
- 自動引用計數
- 可選鏈
- 錯誤處理
- 類型轉換
- 嵌套類型
- 擴展
- 協議
- 泛型
- 權限控制
- 高級操作符
- 語言參考
- 關于語言參考
- 詞法結構
- 類型
- 表達式
- 語句
- 聲明
- 特性
- 模式
- 泛型參數
- 語法總結
- 蘋果官方Blog官方翻譯
- Access Control 權限控制的黑與白
- 造個類型不是夢-白話Swift類型創建
- WWDC里面的那個“大炮打氣球”
- Swift與C語言指針友好合作
- 引用類型和值類型的恩怨
- 訪問控制和Protected
- 可選類型完美解決占位問題