Swift 語言提供Arrays
潮罪、Sets
和Dictionaries
三種基本的集合類型用來存儲(chǔ)集合數(shù)據(jù)铐刘。數(shù)組(Arrays)是有序數(shù)據(jù)的集敞临。集合(Sets)是無序無重復(fù)數(shù)據(jù)的集妨猩。字典(Dictionaries)是無序的鍵值對(duì)的集八匠。
Swift 語言中的Arrays
捂襟、Sets
和Dictionaries
中存儲(chǔ)的數(shù)據(jù)值類型必須明確咬腕。這意味著我們不能把不正確的數(shù)據(jù)類型插入其中。同時(shí)這也說明我們完全可以對(duì)取回值的類型非常自信葬荷。
注意:
Swift 的Arrays
涨共、Sets
和Dictionaries
類型被實(shí)現(xiàn)為泛型集合纽帖。
集合的可變性
如果創(chuàng)建一個(gè)Arrays
、Sets
或Dictionaries
并且把它分配成一個(gè)變量煞赢,這個(gè)集合將會(huì)是可變的抛计。這意味著我們可以在創(chuàng)建之后添加更多或移除已存在的數(shù)據(jù)項(xiàng),或者改變集合中的數(shù)據(jù)項(xiàng)照筑。如果我們把Arrays
吹截、Sets
或Dictionaries
分配成常量,那么它就是不可變的凝危,它的大小和內(nèi)容都不能被改變波俄。
注意:
在我們不需要改變集合的時(shí)候創(chuàng)建不可變集合是很好的實(shí)踐。如此 Swift 編譯器可以優(yōu)化我們創(chuàng)建的集合蛾默。
數(shù)組(Arrays)
數(shù)組使用有序列表存儲(chǔ)同一類型的多個(gè)值懦铺。相同的值可以多次出現(xiàn)在一個(gè)數(shù)組的不同位置中。
注意:
Swift 的Array
類型被橋接到Foundation
中的NSArray
類支鸡。
數(shù)組的簡單語法
寫 Swift 數(shù)組應(yīng)該遵循像Array<Element>
這樣的形式冬念,其中Element
是這個(gè)數(shù)組中唯一允許存在的數(shù)據(jù)類型。我們也可以使用像[Element]
這樣的簡單語法牧挣。盡管兩種形式在功能上是一樣的急前,但是推薦較短的那種,而且在本文中都會(huì)使用這種形式來使用數(shù)組瀑构。
創(chuàng)建一個(gè)空數(shù)組
我們可以使用構(gòu)造語法來創(chuàng)建一個(gè)由特定數(shù)據(jù)類型構(gòu)成的空數(shù)組:
var someInts = [Int]()
print("someInts is of type [Int] with \(someInts.count) items.")
// 打印 "someInts is of type [Int] with 0 items."
注意裆针,通過構(gòu)造函數(shù)的類型,someInts
的值類型被推斷為[Int]
寺晌。
或者世吨,如果代碼上下文中已經(jīng)提供了類型信息,例如一個(gè)函數(shù)參數(shù)或者一個(gè)已經(jīng)定義好類型的常量或者變量呻征,我們可以使用空數(shù)組語句創(chuàng)建一個(gè)空數(shù)組耘婚,它的寫法很簡單:[]
(一對(duì)空方括號(hào)):
someInts.append(3)
// someInts 現(xiàn)在包含一個(gè) Int 值
someInts = []
// someInts 現(xiàn)在是空數(shù)組,但是仍然是 [Int] 類型的陆赋。
創(chuàng)建一個(gè)帶有默認(rèn)值的數(shù)組
Swift 中的Array
類型還提供一個(gè)可以創(chuàng)建特定大小并且所有數(shù)據(jù)都被默認(rèn)的構(gòu)造方法边篮。我們可以把準(zhǔn)備加入新數(shù)組的數(shù)據(jù)項(xiàng)數(shù)量(count
)和適當(dāng)類型的初始值(repeatedValue
)傳入數(shù)組構(gòu)造函數(shù):
var threeDoubles = [Double](count: 3, repeatedValue:0.0)
// threeDoubles 是一種 [Double] 數(shù)組,等價(jià)于 [0.0, 0.0, 0.0]
通過兩個(gè)數(shù)組相加創(chuàng)建一個(gè)數(shù)組
我們可以使用加法操作符(+
)來組合兩種已存在的相同類型數(shù)組奏甫。新數(shù)組的數(shù)據(jù)類型會(huì)被從兩個(gè)數(shù)組的數(shù)據(jù)類型中推斷出來:
var anotherThreeDoubles = Array(count: 3, repeatedValue: 2.5)
// anotherThreeDoubles 被推斷為 [Double],等價(jià)于 [2.5, 2.5, 2.5]
var sixDoubles = threeDoubles + anotherThreeDoubles
// sixDoubles 被推斷為 [Double]凌受,等價(jià)于 [0.0, 0.0, 0.0, 2.5, 2.5, 2.5]
用字面量構(gòu)造數(shù)組
我們可以使用字面量來進(jìn)行數(shù)組構(gòu)造阵子,這是一種用一個(gè)或者多個(gè)數(shù)值構(gòu)造數(shù)組的簡單方法。字面量是一系列由逗號(hào)分割并由方括號(hào)包含的數(shù)值:
[value 1, value 2, value 3]
胜蛉。
下面這個(gè)例子創(chuàng)建了一個(gè)叫做shoppingList
并且存儲(chǔ)String
的數(shù)組:
var shoppingList: [String] = ["Eggs", "Milk"]
// shoppingList 已經(jīng)被構(gòu)造并且擁有兩個(gè)初始項(xiàng)挠进。
shoppingList
變量被聲明為“字符串值類型的數(shù)組“色乾,記作[String]
。 因?yàn)檫@個(gè)數(shù)組被規(guī)定只有String
一種數(shù)據(jù)結(jié)構(gòu)领突,所以只有String
類型可以在其中被存取暖璧。 在這里,shoppinglist
數(shù)組由兩個(gè)String
值("Eggs"
和"Milk"
)構(gòu)造君旦,并且由字面量定義澎办。
注意:
Shoppinglist
數(shù)組被聲明為變量(var
關(guān)鍵字創(chuàng)建)而不是常量(let
創(chuàng)建)是因?yàn)橐院罂赡軙?huì)有更多的數(shù)據(jù)項(xiàng)被插入其中。
在這個(gè)例子中金砍,字面量僅僅包含兩個(gè)String
值局蚀。匹配了該數(shù)組的變量聲明(只能包含String
的數(shù)組),所以這個(gè)字面量的分配過程可以作為用兩個(gè)初始項(xiàng)來構(gòu)造shoppinglist
的一種方式恕稠。
由于 Swift 的類型推斷機(jī)制琅绅,當(dāng)我們用字面量構(gòu)造只擁有相同類型值數(shù)組的時(shí)候,我們不必把數(shù)組的類型定義清楚鹅巍。 shoppinglist
的構(gòu)造也可以這樣寫:
var shoppingList = ["Eggs", "Milk"]
因?yàn)樗凶置媪恐械闹刀际窍嗤念愋颓Х觯琒wift 可以推斷出[String]
是shoppinglist
中變量的正確類型。
訪問和修改數(shù)組
我們可以通過數(shù)組的方法和屬性來訪問和修改數(shù)組骆捧,或者使用下標(biāo)語法澎羞。
可以使用數(shù)組的只讀屬性count
來獲取數(shù)組中的數(shù)據(jù)項(xiàng)數(shù)量:
print("The shopping list contains \(shoppingList.count) items.")
// 輸出 "The shopping list contains 2 items."(這個(gè)數(shù)組有2個(gè)項(xiàng))
使用布爾值屬性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(_:)
方法在數(shù)組后面添加新的數(shù)據(jù)項(xiàng):
shoppingList.append("Flour")
// shoppingList 現(xiàn)在有3個(gè)數(shù)據(jù)項(xiàng),有人在攤煎餅
除此之外凑懂,使用加法賦值運(yùn)算符(+=
)也可以直接在數(shù)組后面添加一個(gè)或多個(gè)擁有相同類型的數(shù)據(jù)項(xiàng):
shoppingList += ["Baking Powder"]
// shoppingList 現(xiàn)在有四項(xiàng)了
shoppingList += ["Chocolate Spread", "Cheese", "Butter"]
// shoppingList 現(xiàn)在有七項(xiàng)了
可以直接使用下標(biāo)語法來獲取數(shù)組中的數(shù)據(jù)項(xiàng)煤痕,把我們需要的數(shù)據(jù)項(xiàng)的索引值放在直接放在數(shù)組名稱的方括號(hào)中:
var firstItem = shoppingList[0]
// 第一項(xiàng)是 "Eggs"
注意:
第一項(xiàng)在數(shù)組中的索引值是0
而不是1
。 Swift 中的數(shù)組索引總是從零開始接谨。
我們也可以用下標(biāo)來改變某個(gè)已有索引值對(duì)應(yīng)的數(shù)據(jù)值:
shoppingList[0] = "Six eggs"
// 其中的第一項(xiàng)現(xiàn)在是 "Six eggs" 而不是 "Eggs"
還可以利用下標(biāo)來一次改變一系列數(shù)據(jù)值摆碉,即使新數(shù)據(jù)和原有數(shù)據(jù)的數(shù)量是不一樣的。下面的例子把"Chocolate Spread"
脓豪,"Cheese"
巷帝,和"Butter"
替換為"Bananas"
和 "Apples"
:
shoppingList[4...6] = ["Bananas", "Apples"]
// shoppingList 現(xiàn)在有6項(xiàng)
注意:
不可以用下標(biāo)訪問的形式去在數(shù)組尾部添加新項(xiàng)。
調(diào)用數(shù)組的insert(_:atIndex:)
方法來在某個(gè)具體索引值之前添加數(shù)據(jù)項(xiàng):
shoppingList.insert("Maple Syrup", atIndex: 0)
// shoppingList 現(xiàn)在有7項(xiàng)
// "Maple Syrup" 現(xiàn)在是這個(gè)列表中的第一項(xiàng)
這次insert(_:atIndex:)
方法調(diào)用把值為"Maple Syrup"
的新數(shù)據(jù)項(xiàng)插入列表的最開始位置扫夜,并且使用0
作為索引值楞泼。
類似的我們可以使用removeAtIndex(_:)
方法來移除數(shù)組中的某一項(xiàng)。這個(gè)方法把數(shù)組在特定索引值中存儲(chǔ)的數(shù)據(jù)項(xiàng)移除并且返回這個(gè)被移除的數(shù)據(jù)項(xiàng)(我們不需要的時(shí)候就可以無視它):
let mapleSyrup = shoppingList.removeAtIndex(0)
// 索引值為0的數(shù)據(jù)項(xiàng)被移除
// shoppingList 現(xiàn)在只有6項(xiàng)笤闯,而且不包括 Maple Syrup
// mapleSyrup 常量的值等于被移除數(shù)據(jù)項(xiàng)的值 "Maple Syrup"
注意:
如果我們?cè)囍鴮?duì)索引越界的數(shù)據(jù)進(jìn)行檢索或者設(shè)置新值的操作堕阔,會(huì)引發(fā)一個(gè)運(yùn)行時(shí)錯(cuò)誤。我們可以使用索引值和數(shù)組的count
屬性進(jìn)行比較來在使用某個(gè)索引之前先檢驗(yàn)是否有效颗味。除了當(dāng)count
等于 0 時(shí)(說明這是個(gè)空數(shù)組)超陆,最大索引值一直是count - 1
,因?yàn)閿?shù)組都是零起索引浦马。
數(shù)據(jù)項(xiàng)被移除后數(shù)組中的空出項(xiàng)會(huì)被自動(dòng)填補(bǔ)时呀,所以現(xiàn)在索引值為0
的數(shù)據(jù)項(xiàng)的值再次等于"Six eggs"
:
firstItem = shoppingList[0]
// firstItem 現(xiàn)在等于 "Six eggs"
如果我們只想把數(shù)組中的最后一項(xiàng)移除张漂,可以使用removeLast()
方法而不是removeAtIndex(_:)
方法來避免我們需要獲取數(shù)組的count
屬性。就像后者一樣谨娜,前者也會(huì)返回被移除的數(shù)據(jù)項(xiàng):
let apples = shoppingList.removeLast()
// 數(shù)組的最后一項(xiàng)被移除了
// shoppingList 現(xiàn)在只有5項(xiàng)航攒,不包括 cheese
// apples 常量的值現(xiàn)在等于 "Apples" 字符串
數(shù)組的遍歷
我們可以使用for-in
循環(huán)來遍歷所有數(shù)組中的數(shù)據(jù)項(xiàng):
for item in shoppingList {
print(item)
}
// Six eggs
// Milk
// Flour
// Baking Powder
// Bananas
如果我們同時(shí)需要每個(gè)數(shù)據(jù)項(xiàng)的值和索引值,可以使用enumerate()
方法來進(jìn)行數(shù)組遍歷趴梢。enumerate()
返回一個(gè)由每一個(gè)數(shù)據(jù)項(xiàng)索引值和數(shù)據(jù)值組成的元組漠畜。我們可以把這個(gè)元組分解成臨時(shí)常量或者變量來進(jìn)行遍歷:
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
集合(Sets)
集合(Set)用來存儲(chǔ)相同類型并且沒有確定順序的值。當(dāng)集合元素順序不重要時(shí)或者希望確保每個(gè)元素只出現(xiàn)一次時(shí)可以使用集合而不是數(shù)組垢油。
注意:
Swift的Set
類型被橋接到Foundation
中的NSSet
類盆驹。
集合類型的哈希值
一個(gè)類型為了存儲(chǔ)在集合中,該類型必須是可哈咸渤睿化的--也就是說躯喇,該類型必須提供一個(gè)方法來計(jì)算它的哈希值。一個(gè)哈希值是Int
類型的硝枉,相等的對(duì)象哈希值必須相同廉丽,比如a==b
,因此必須a.hashValue == b.hashValue
。
Swift 的所有基本類型(比如String
,Int
,Double
和Bool
)默認(rèn)都是可哈掀尬叮化的正压,可以作為集合的值的類型或者字典的鍵的類型。沒有關(guān)聯(lián)值的枚舉成員值默認(rèn)也是可哈显鹎颍化的焦履。
注意:
可以使用自定義的類型作為集合的值的類型或者是字典的鍵的類型,但需要使自定義類型符合 Swift 標(biāo)準(zhǔn)庫中的Hashable
協(xié)議雏逾。符合Hashable
協(xié)議的類型需要提供一個(gè)類型為Int
的可讀屬性hashValue
嘉裤。由類型的hashValue
屬性返回的值不需要在同一程序的不同執(zhí)行周期或者不同程序之間保持相同。
因?yàn)?code>Hashable協(xié)議符合
Equatable
協(xié)議栖博,所以符合該協(xié)議的類型也必須提供一個(gè)"是否相等"運(yùn)算符(==
)的實(shí)現(xiàn)屑宠。這個(gè)Equatable
協(xié)議要求任何符合==
實(shí)現(xiàn)的實(shí)例間都是一種相等的關(guān)系。也就是說仇让,對(duì)于a,b,c
三個(gè)值來說典奉,==
的實(shí)現(xiàn)必須滿足下面三種情況:
a == a
(自反性)a == b
意味著b == a
(對(duì)稱性)a == b && b == c
意味著a == c
(傳遞性)
集合類型語法
Swift 中的Set
類型被寫為Set<Element>
,這里的Element
表示Set
中允許存儲(chǔ)的類型丧叽,和數(shù)組不同的是卫玖,集合沒有等價(jià)的簡化形式。
創(chuàng)建和構(gòu)造一個(gè)空的集合
可以通過構(gòu)造器語法創(chuàng)建一個(gè)特定類型的空集合:
var letters = Set<Character>()
print("letters is of type Set<Character> with \(letters.count) items.")
// 打印 "letters is of type Set<Character> with 0 items."
注意:
通過構(gòu)造器踊淳,這里的letters
變量的類型被推斷為Set<Character>
假瞬。
此外,如果上下文提供了類型信息,比如作為函數(shù)的參數(shù)或者已知類型的變量或常量笨触,我們可以通過一個(gè)空的數(shù)組字面量創(chuàng)建一個(gè)空的Set
:
letters.insert("a")
// letters 現(xiàn)在含有1個(gè) Character 類型的值
letters = []
// letters 現(xiàn)在是一個(gè)空的 Set, 但是它依然是 Set<Character> 類型
用數(shù)組字面量創(chuàng)建集合
可以使用數(shù)組字面量來構(gòu)造集合,并且可以使用簡化形式寫一個(gè)或者多個(gè)值作為集合元素雹舀。
下面的例子創(chuàng)建一個(gè)稱之為favoriteGenres
的集合來存儲(chǔ)String
類型的值:
var favoriteGenres: Set<String> = ["Rock", "Classical", "Hip hop"]
// favoriteGenres 被構(gòu)造成含有三個(gè)初始值的集合
這個(gè)favoriteGenres
變量被聲明為“一個(gè)String
值的集合”芦劣,寫為Set<String>
。由于這個(gè)特定的集合含有指定String
類型的值说榆,所以它只允許存儲(chǔ)String
類型值虚吟。這里的favoriteGenres
變量有三個(gè)String
類型的初始值("Rock"
,"Classical"
和"Hip hop"
)签财,并以數(shù)組字面量的方式出現(xiàn)串慰。
注意:
favoriteGenres
被聲明為一個(gè)變量(擁有var
標(biāo)示符)而不是一個(gè)常量(擁有let
標(biāo)示符),因?yàn)樗锩娴脑貙?huì)在下面的例子中被增加或者移除。
一個(gè)Set
類型不能從數(shù)組字面量中被單獨(dú)推斷出來唱蒸,因此Set
類型必須顯式聲明邦鲫。然而,由于 Swift 的類型推斷功能神汹,如果想使用一個(gè)數(shù)組字面量構(gòu)造一個(gè)Set
并且該數(shù)組字面量中的所有元素類型相同庆捺,那么無須寫出Set
的具體類型。favoriteGenres
的構(gòu)造形式可以采用簡化的方式代替:
var favoriteGenres: Set = ["Rock", "Classical", "Hip hop"]
由于數(shù)組字面量中的所有元素類型相同屁魏,Swift 可以推斷出Set<String>
作為favoriteGenres
變量的正確類型滔以。
訪問和修改一個(gè)集合
可以通過Set
的屬性和方法來訪問和修改一個(gè)Set
。
為了找出一個(gè)Set
中元素的數(shù)量氓拼,可以使用其只讀屬性count
:
print("I have \(favoriteGenres.count) favorite music genres.")
// 打印 "I have 3 favorite music genres."
使用布爾屬性isEmpty
作為一個(gè)縮寫形式去檢查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."
可以通過調(diào)用Set
的insert(_:)
方法來添加一個(gè)新元素:
favoriteGenres.insert("Jazz")
// favoriteGenres 現(xiàn)在包含4個(gè)元素
可以通過調(diào)用Set
的remove(_:)
方法去刪除一個(gè)元素你画,如果該值是該Set
的一個(gè)元素則刪除該元素并且返回被刪除的元素值,否則如果該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
中是否包含一個(gè)特定的值:
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."
遍歷一個(gè)集合
可以在一個(gè)for-in
循環(huán)中遍歷一個(gè)Set
中的所有值剥槐。
for genre in favoriteGenres {
print("\(genre)")
}
// Classical
// Jazz
// Hip hop
Swift 的Set
類型沒有確定的順序,為了按照特定順序來遍歷一個(gè)Set
中的值可以使用sort()
方法宪摧,它將根據(jù)提供的序列返回一個(gè)有序集合.
for genre in favoriteGenres.sort() {
print("\(genre)")
}
// prints "Classical"
// prints "Hip hop"
// prints "Jazz
集合操作
可以高效地完成Set
的一些基本操作粒竖,比如把兩個(gè)集合組合到一起,判斷兩個(gè)集合共有元素几于,或者判斷兩個(gè)集合是否全包含蕊苗,部分包含或者不相交。
基本集合操作
下面的插圖描述了兩個(gè)集合-a
和b
-以及通過陰影部分的區(qū)域顯示集合各種操作的結(jié)果沿彭。
- 使用
intersect(_:)
方法根據(jù)兩個(gè)集合中都包含的值創(chuàng)建的一個(gè)新的集合朽砰。 - 使用
exclusiveOr(_:)
方法根據(jù)在一個(gè)集合中但不在兩個(gè)集合中的值創(chuàng)建一個(gè)新的集合。 - 使用
union(_:)
方法根據(jù)兩個(gè)集合的值創(chuàng)建一個(gè)新的集合。 - 使用
subtract(_:)
方法根據(jù)不在該集合中的值創(chuàng)建一個(gè)新的集合瞧柔。
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]
集合成員關(guān)系和相等
下面的插圖描述了三個(gè)集合-a
,b
和c
,以及通過重疊區(qū)域表述集合間共享的元素漆弄。集合a
是集合b
的父集合,因?yàn)?code>a包含了b
中所有的元素造锅,相反的撼唾,集合b
是集合a
的子集合,因?yàn)閷儆?code>b的元素也被a
包含哥蔚。集合b
和集合c
彼此不關(guān)聯(lián)倒谷,因?yàn)樗鼈冎g沒有共同的元素。
- 使用“是否相等”運(yùn)算符(
==
)來判斷兩個(gè)集合是否包含全部相同的值糙箍。 - 使用
isSubsetOf(_:)
方法來判斷一個(gè)集合中的值是否也被包含在另外一個(gè)集合中渤愁。 - 使用
isSupersetOf(_:)
方法來判斷一個(gè)集合中包含另一個(gè)集合中所有的值。 - 使用
isStrictSubsetOf(_:)
或者isStrictSupersetOf(_:)
方法來判斷一個(gè)集合是否是另外一個(gè)集合的子集合或者父集合并且兩個(gè)集合并不相等深夯。 - 使用
isDisjointWith(_:)
方法來判斷兩個(gè)集合是否不含有相同的值抖格。
let houseAnimals: Set = ["??", "??"]
let farmAnimals: Set = ["??", "??", "??", "??", "??"]
let cityAnimals: Set = ["??", "??"]
houseAnimals.isSubsetOf(farmAnimals)
// true
farmAnimals.isSupersetOf(houseAnimals)
// true
farmAnimals.isDisjointWith(cityAnimals)
// true
字典
字典是一種存儲(chǔ)多個(gè)相同類型的值的容器。每個(gè)值(value)都關(guān)聯(lián)唯一的鍵(key)塌西,鍵作為字典中的這個(gè)值數(shù)據(jù)的標(biāo)識(shí)符他挎。和數(shù)組中的數(shù)據(jù)項(xiàng)不同,字典中的數(shù)據(jù)項(xiàng)并沒有具體順序捡需。我們?cè)谛枰ㄟ^標(biāo)識(shí)符(鍵)訪問數(shù)據(jù)的時(shí)候使用字典办桨,這種方法很大程度上和我們?cè)诂F(xiàn)實(shí)世界中使用字典查字義的方法一樣。
注意:
Swift 的Dictionary
類型被橋接到Foundation
的NSDictionary
類站辉。
字典類型快捷語法
Swift 的字典使用Dictionary<Key, Value>
定義呢撞,其中Key
是字典中鍵的數(shù)據(jù)類型,Value
是字典中對(duì)應(yīng)于這些鍵所存儲(chǔ)值的數(shù)據(jù)類型饰剥。
注意:
一個(gè)字典的Key
類型必須遵循Hashable
協(xié)議殊霞,就像Set
的值類型。
我們也可以用[Key: Value]
這樣快捷的形式去創(chuàng)建一個(gè)字典類型汰蓉。雖然這兩種形式功能上相同绷蹲,但是后者是首選,并且涉及到字典類型時(shí)通篇采用后者顾孽。
創(chuàng)建一個(gè)空字典
我們可以像數(shù)組一樣使用構(gòu)造語法創(chuàng)建一個(gè)擁有確定類型的空字典:
var namesOfIntegers = [Int: String]()
// namesOfIntegers 是一個(gè)空的 [Int: String] 字典
這個(gè)例子創(chuàng)建了一個(gè)[Int: String]
類型的空字典來儲(chǔ)存整數(shù)的英語命名祝钢。它的鍵是Int
型,值是String
型若厚。
如果上下文已經(jīng)提供了類型信息拦英,我們可以使用空字典字面量來創(chuàng)建一個(gè)空字典,記作[:]
(中括號(hào)中放一個(gè)冒號(hào)):
namesOfIntegers[16] = "sixteen"
// namesOfIntegers 現(xiàn)在包含一個(gè)鍵值對(duì)
namesOfIntegers = [:]
// namesOfIntegers 又成為了一個(gè) [Int: String] 類型的空字典
用字典字面量創(chuàng)建字典
我們可以使用字典字面量來構(gòu)造字典测秸,這和我們剛才介紹過的數(shù)組字面量擁有相似語法疤估。字典字面量是一種將一個(gè)或多個(gè)鍵值對(duì)寫作Dictionary
集合的快捷途徑灾常。
一個(gè)鍵值對(duì)是一個(gè)key
和一個(gè)value
的結(jié)合體。在字典字面量中铃拇,每一個(gè)鍵值對(duì)的鍵和值都由冒號(hào)分割钞瀑。這些鍵值對(duì)構(gòu)成一個(gè)列表,其中這些鍵值對(duì)由方括號(hào)包含慷荔、由逗號(hào)分割:
[key 1: value 1, key 2: value 2, key 3: value 3]
下面的例子創(chuàng)建了一個(gè)存儲(chǔ)國際機(jī)場名稱的字典仔戈。在這個(gè)字典中鍵是三個(gè)字母的國際航空運(yùn)輸相關(guān)代碼,值是機(jī)場名稱:
var airports: [String: String] = ["YYZ": "Toronto Pearson", "DUB": "Dublin"]
airports
字典被聲明為一種[String: String]
類型拧廊,這意味著這個(gè)字典的鍵和值都是String
類型。
注意:
airports
字典被聲明為變量(用var
關(guān)鍵字)而不是常量(let
關(guān)鍵字)因?yàn)楹髞砀嗟臋C(jī)場信息會(huì)被添加到這個(gè)示例字典中晋修。
airports
字典使用字典字面量初始化吧碾,包含兩個(gè)鍵值對(duì)。第一對(duì)的鍵是YYZ
墓卦,值是Toronto Pearson
倦春。第二對(duì)的鍵是DUB
,值是Dublin
落剪。
這個(gè)字典語句包含了兩個(gè)String: String
類型的鍵值對(duì)睁本。它們對(duì)應(yīng)airports
變量聲明的類型(一個(gè)只有String
鍵和String
值的字典)所以這個(gè)字典字面量的任務(wù)是構(gòu)造擁有兩個(gè)初始數(shù)據(jù)項(xiàng)的airport
字典。
和數(shù)組一樣忠怖,我們?cè)谟米值渥置媪繕?gòu)造字典時(shí)呢堰,如果它的鍵和值都有各自一致的類型,那么就不必寫出字典的類型凡泣。
airports
字典也可以用這種簡短方式定義:
var airports = ["YYZ": "Toronto Pearson", "DUB": "Dublin"]
因?yàn)檫@個(gè)語句中所有的鍵和值都各自擁有相同的數(shù)據(jù)類型枉疼,Swift 可以推斷出Dictionary<String, String>
是airports
字典的正確類型。
訪問和修改字典
我們可以通過字典的方法和屬性來訪問和修改字典鞋拟,或者通過使用下標(biāo)語法骂维。
和數(shù)組一樣,我們可以通過字典的只讀屬性count
來獲取某個(gè)字典的數(shù)據(jù)項(xiàng)數(shù)量:
print("The dictionary of airports contains \(airports.count) items.")
// 打印 "The dictionary of airports contains 2 items."(這個(gè)字典有兩個(gè)數(shù)據(jù)項(xiàng))
使用布爾屬性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."
我們也可以在字典中使用下標(biāo)語法來添加新的數(shù)據(jù)項(xiàng)贺纲『焦耄可以使用一個(gè)恰當(dāng)類型的鍵作為下標(biāo)索引,并且分配恰當(dāng)類型的新值:
airports["LHR"] = "London"
// airports 字典現(xiàn)在有三個(gè)數(shù)據(jù)項(xiàng)
我們也可以使用下標(biāo)語法來改變特定鍵對(duì)應(yīng)的值:
airports["LHR"] = "London Heathrow"
// "LHR"對(duì)應(yīng)的值 被改為 "London Heathrow
作為另一種下標(biāo)方法猴誊,字典的updateValue(_:forKey:)
方法可以設(shè)置或者更新特定鍵對(duì)應(yīng)的值潦刃。就像上面所示的下標(biāo)示例,updateValue(_:forKey:)
方法在這個(gè)鍵不存在對(duì)應(yīng)值的時(shí)候會(huì)設(shè)置新值或者在存在時(shí)更新已存在的值稠肘。和上面的下標(biāo)方法不同的福铅,updateValue(_:forKey:)
這個(gè)方法返回更新值之前的原值。這樣使得我們可以檢查更新是否成功项阴。
updateValue(_:forKey:)
方法會(huì)返回對(duì)應(yīng)值的類型的可選值滑黔。舉例來說:對(duì)于存儲(chǔ)String
值的字典笆包,這個(gè)函數(shù)會(huì)返回一個(gè)String?
或者“可選 String
”類型的值。
如果有值存在于更新前略荡,則這個(gè)可選值包含了舊值庵佣,否則它將會(huì)是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."
我們也可以使用下標(biāo)語法來在字典中檢索特定鍵對(duì)應(yīng)的值汛兜。因?yàn)橛锌赡苷?qǐng)求的鍵沒有對(duì)應(yīng)的值存在巴粪,字典的下標(biāo)訪問會(huì)返回對(duì)應(yīng)值的類型的可選值。如果這個(gè)字典包含請(qǐng)求鍵所對(duì)應(yīng)的值粥谬,下標(biāo)會(huì)返回一個(gè)包含這個(gè)存在值的可選值肛根,否則將返回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."
我們還可以使用下標(biāo)語法來通過給某個(gè)鍵的對(duì)應(yīng)值賦值為nil
來從字典里移除一個(gè)鍵值對(duì):
airports["APL"] = "Apple Internation"
// "Apple Internation" 不是真的 APL 機(jī)場, 刪除它
airports["APL"] = nil
// APL 現(xiàn)在被移除了
此外,removeValueForKey(_:)
方法也可以用來在字典中移除鍵值對(duì)漏策。這個(gè)方法在鍵值對(duì)存在的情況下會(huì)移除該鍵值對(duì)并且返回被移除的值或者在沒有值的情況下返回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
循環(huán)來遍歷某個(gè)字典中的鍵值對(duì)派哲。每一個(gè)字典中的數(shù)據(jù)項(xiàng)都以(key, value)
元組形式返回,并且我們可以使用臨時(shí)常量或者變量來分解這些元組:
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
如果我們只是需要使用某個(gè)字典的鍵集合或者值集合來作為某個(gè)接受Array
實(shí)例的 API 的參數(shù)芭届,可以直接使用keys
或者values
屬性構(gòu)造一個(gè)新數(shù)組:
let airportCodes = [String](airports.keys)
// airportCodes 是 ["YYZ", "LHR"]
let airportNames = [String](airports.values)
// airportNames 是 ["Toronto Pearson", "London Heathrow"]
Swift 的字典類型是無序集合類型。為了以特定的順序遍歷字典的鍵或值感耙,可以對(duì)字典的keys
或values
屬性使用sort()
方法褂乍。