4.集合類型[Collection Types]

[The Swift Programming Language 中文版]
本頁包含內(nèi)容:

集合的可變性(Mutability of Collections)
數(shù)組(Arrays)
集合(Sets)
字典(Dictionaries)

Swift 語言提供Arrays、Sets和Dictionaries三種基本的集合類型用來存儲集合數(shù)據(jù)屈尼。數(shù)組(Arrays)是有序數(shù)據(jù)的集瞻讽。集合(Sets)是無序無重復(fù)數(shù)據(jù)的集。字典(Dictionaries)是無序的鍵值對的集。

Swift 語言中的Arrays、Sets和Dictionaries中存儲的數(shù)據(jù)值類型必須明確。這意味著我們不能把不正確的數(shù)據(jù)類型插入其中槽地。同時(shí)這也說明我們完全可以對取回值的類型非常自信。

注意:
Swift 的Arrays芦瘾、Sets和Dictionaries類型被實(shí)現(xiàn)為泛型集合捌蚊。更多關(guān)于泛型類型和集合,參見 泛型章節(jié)近弟。

集合的可變性
如果創(chuàng)建一個(gè)Arrays缅糟、Sets或Dictionaries并且把它分配成一個(gè)變量,這個(gè)集合將會是可變的祷愉。這意味著我們可以在創(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ù)組使用有序列表存儲同一類型的多個(gè)值欺殿。相同的值可以多次出現(xiàn)在一個(gè)數(shù)組的不同位置中寄纵。

注意: Swift 的Array類型被橋接到Foundation中的NSArray類。 更多關(guān)于在Foundation和Cocoa中使用Array的信息脖苏,參見 Using Swift with Cocoa and Obejective-C 一書程拭。

數(shù)組的簡單語法

寫 Swift 數(shù)組應(yīng)該遵循像Array<Element>這樣的形式,其中Element是這個(gè)數(shù)組中唯一允許存在的數(shù)據(jù)類型棍潘。我們也可以使用像[Element]這樣的簡單語法恃鞋。盡管兩種形式在功能上是一樣的屋吨,但是推薦較短的那種,而且在本文中都會使用這種形式來使用數(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ù)組,它的寫法很簡單:[](一對空方括號):

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ù)類型會被從兩個(gè)數(shù)組的數(shù)據(jù)類型中推斷出來:

var anotherThreeDoubles = [Double](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ù)組的簡單方法。字面量是一系列由逗號分割并由方括號包含的數(shù)值:

[value 1, value 2, value 3]挥唠。

下面這個(gè)例子創(chuàng)建了一個(gè)叫做shoppingList并且存儲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)橐院罂赡軙懈嗟臄?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ù)組名稱的方括號中:

var firstItem = shoppingList[0]
// 第一項(xiàng)是 "Eggs"

注意:
第一項(xiàng)在數(shù)組中的索引值是0而不是1锄贷。 Swift 中的數(shù)組索引總是從零開始。
我們也可以用下標(biāo)來改變某個(gè)已有索引值對應(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ù)組在特定索引值中存儲的數(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"

注意:
如果我們試著對索引越界的數(shù)據(jù)進(jìn)行檢索或者設(shè)置新值的操作末购,會引發(fā)一個(gè)運(yùn)行期錯(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)會被自動填補(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屬性。就像后者一樣迅办,前者也會返回被移除的數(shù)據(jù)項(xiàng):

let apples = shoppingList.removeLast()
// 數(shù)組的最后一項(xiàng)被移除了
// shoppingList 現(xiàn)在只有5項(xiàng)玩般,不包括 Apples
// 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

更多關(guān)于for-in循環(huán)的介紹請參見for 循環(huán)究驴。

集合(Sets)
集合(Set)用來存儲相同類型并且沒有確定順序的值。當(dāng)集合元素順序不重要時(shí)或者希望確保每個(gè)元素只出現(xiàn)一次時(shí)可以使用集合而不是數(shù)組匀伏。

注意:
Swift的Set類型被橋接到Foundation中的NSSet類洒忧。
關(guān)于使用Foundation和Cocoa中Set的知識,請看 Using Swift with Cocoa and Objective-C够颠。

集合類型的哈希值

一個(gè)類型為了存儲在集合中熙侍,該類型必須是可哈希化的--也就是說履磨,該類型必須提供一個(gè)方法來計(jì)算它的哈希值蛉抓。一個(gè)哈希值是Int類型的,相等的對象哈希值必須相同剃诅,比如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)镠ashable協(xié)議符合Equatable協(xié)議,所以符合該協(xié)議的類型也必須提供一個(gè)"是否相等"運(yùn)算符(==)的實(shí)現(xiàn)伸眶。這個(gè)Equatable協(xié)議要求任何符合==實(shí)現(xiàn)的實(shí)例間都是一種相等的關(guān)系芥颈。也就是說,對于a,b,c三個(gè)值來說赚抡,==的實(shí)現(xiàn)必須滿足下面三種情況:

a == a(自反性)
a == b意味著b == a(對稱性)
a == b && b == c意味著a == c(傳遞性)
關(guān)于符合協(xié)議的更多信息爬坑,請看協(xié)議。

集合類型語法

Swift 中的Set類型被寫為Set<Element>涂臣,這里的Element表示Set中允許存儲的類型盾计,和數(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的集合來存儲String類型的值:

var favoriteGenres: Set<String> = ["Rock", "Classical", "Hip hop"]
// favoriteGenres 被構(gòu)造成含有三個(gè)初始值的集合

這個(gè)favoriteGenres變量被聲明為“一個(gè)String值的集合”逝淹,寫為Set<String>。由于這個(gè)特定的集合含有指定String類型的值桶唐,所以它只允許存儲String類型值栅葡。這里的favoriteGenres變量有三個(gè)String類型的初始值("Rock","Classical"和"Hip hop")尤泽,并以數(shù)組字面量的方式出現(xiàn)欣簇。

注意:
favoriteGenres被聲明為一個(gè)變量(擁有var標(biāo)示符)而不是一個(gè)常量(擁有l(wèi)et標(biāo)示符),因?yàn)樗锩娴脑貙谙旅娴睦又斜辉黾踊蛘咭瞥?br> 一個(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

更多關(guān)于for-in循環(huán)的信息,參見For 循環(huán)尚猿。

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é)果。

Set.png

使用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)閍包含了b中所有的元素,相反的帆竹,集合b是集合a的子集合绕娘,因?yàn)閷儆赽的元素也被a包含。集合b和集合c彼此不關(guān)聯(lián)栽连,因?yàn)樗鼈冎g沒有共同的元素险领。

Set.png

使用“是否相等”運(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

字典
字典是一種存儲多個(gè)相同類型的值的容器脐湾。每個(gè)值(value)都關(guān)聯(lián)唯一的鍵(key),鍵作為字典中的這個(gè)值數(shù)據(jù)的標(biāo)識符叙淌。和數(shù)組中的數(shù)據(jù)項(xiàng)不同秤掌,字典中的數(shù)據(jù)項(xiàng)并沒有具體順序。我們在需要通過標(biāo)識符(鍵)訪問數(shù)據(jù)的時(shí)候使用字典鹰霍,這種方法很大程度上和我們在現(xiàn)實(shí)世界中使用字典查字義的方法一樣闻鉴。

注意:
Swift 的Dictionary類型被橋接到Foundation的NSDictionary類。
更多關(guān)于在Foundation和Cocoa中使用Dictionary類型的信息茂洒,參見 Using Swift with Cocoa and Objective-C (Swift 2.1) 一書孟岛。

字典類型快捷語法
Swift 的字典使用Dictionary<Key, Value>定義,其中Key是字典中鍵的數(shù)據(jù)類型督勺,Value是字典中對應(yīng)于這些鍵所存儲值的數(shù)據(jù)類型渠羞。

注意:
一個(gè)字典的Key類型必須遵循Hashable協(xié)議,就像Set的值類型智哀。
我們也可以用[Key: Value]這樣快捷的形式去創(chuàng)建一個(gè)字典類型次询。雖然這兩種形式功能上相同,但是后者是首選瓷叫,并且這本指導(dǎo)書涉及到字典類型時(shí)通篇采用后者屯吊。

創(chuàng)建一個(gè)空字典

我們可以像數(shù)組一樣使用構(gòu)造語法創(chuàng)建一個(gè)擁有確定類型的空字典:

var namesOfIntegers = [Int: String]()
// namesOfIntegers 是一個(gè)空的 [Int: String] 字典

這個(gè)例子創(chuàng)建了一個(gè)[Int: String]類型的空字典來儲存整數(shù)的英語命名。它的鍵是Int型赞辩,值是String型雌芽。

如果上下文已經(jīng)提供了類型信息,我們可以使用空字典字面量來創(chuàng)建一個(gè)空字典辨嗽,記作[:](中括號中放一個(gè)冒號):

namesOfIntegers[16] = "sixteen"
// namesOfIntegers 現(xiàn)在包含一個(gè)鍵值對
namesOfIntegers = [:]
// namesOfIntegers 又成為了一個(gè) [Int: String] 類型的空字典

用字典字面量創(chuàng)建字典
我們可以使用字典字面量來構(gòu)造字典世落,這和我們剛才介紹過的數(shù)組字面量擁有相似語法。字典字面量是一種將一個(gè)或多個(gè)鍵值對寫作Dictionary集合的快捷途徑。

一個(gè)鍵值對是一個(gè)key和一個(gè)value的結(jié)合體屉佳。在字典字面量中谷朝,每一個(gè)鍵值對的鍵和值都由冒號分割。這些鍵值對構(gòu)成一個(gè)列表武花,其中這些鍵值對由方括號包含圆凰、由逗號分割:

[key 1: value 1, key 2: value 2, key 3: value 3]

下面的例子創(chuàng)建了一個(gè)存儲國際機(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ī)場信息會被添加到這個(gè)示例字典中累铅。
airports字典使用字典字面量初始化跃须,包含兩個(gè)鍵值對。第一對的鍵是YYZ娃兽,值是Toronto Pearson菇民。第二對的鍵是DUB,值是Dublin投储。

這個(gè)字典語句包含了兩個(gè)String: String類型的鍵值對第练。它們對應(yīng)airports變量聲明的類型(一個(gè)只有String鍵和String值的字典)所以這個(gè)字典字面量的任務(wù)是構(gòu)造擁有兩個(gè)初始數(shù)據(jù)項(xiàng)的airport字典榜贴。

和數(shù)組一樣竭讳,我們在用字典字面量構(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)語法來改變特定鍵對應(yīng)的值:

airports["LHR"] = "London Heathrow"
// "LHR"對應(yīng)的值 被改為 "London Heathrow
作為另一種下標(biāo)方法闽巩,字典的updateValue(:forKey:)方法可以設(shè)置或者更新特定鍵對應(yīng)的值。就像上面所示的下標(biāo)示例担汤,updateValue(:forKey:)方法在這個(gè)鍵不存在對應(yīng)值的時(shí)候會設(shè)置新值或者在存在時(shí)更新已存在的值涎跨。和上面的下標(biāo)方法不同的,updateValue(_:forKey:)這個(gè)方法返回更新值之前的原值崭歧。這樣使得我們可以檢查更新是否成功隅很。

updateValue(_:forKey:)方法會返回對應(yīng)值的類型的可選值。舉例來說:對于存儲String值的字典率碾,這個(gè)函數(shù)會返回一個(gè)String?或者“可選 String”類型的值叔营。

如果有值存在于更新前屋彪,則這個(gè)可選值包含了舊值,否則它將會是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)語法來在字典中檢索特定鍵對應(yīng)的值畜挥。因?yàn)橛锌赡苷埱蟮逆I沒有對應(yīng)的值存在,字典的下標(biāo)訪問會返回對應(yīng)值的類型的可選值婴谱。如果這個(gè)字典包含請求鍵所對應(yīng)的值蟹但,下標(biāo)會返回一個(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è)鍵的對應(yīng)值賦值為nil來從字典里移除一個(gè)鍵值對:

airports["APL"] = "Apple Internation"
// "Apple Internation" 不是真的 APL 機(jī)場, 刪除它
airports["APL"] = nil
// APL 現(xiàn)在被移除了

此外谭羔,removeValueForKey(_:)方法也可以用來在字典中移除鍵值對矮湘。這個(gè)方法在鍵值對存在的情況下會移除該鍵值對并且返回被移除的值或者在沒有值的情況下返回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è)字典中的鍵值對。每一個(gè)字典中的數(shù)據(jù)項(xiàng)都以(key, value)元組形式返回口糕,并且我們可以使用臨時(shí)常量或者變量來分解這些元組:

for (airportCode, airportName) in airports {
    print("\(airportCode): \(airportName)")
}
// YYZ: Toronto Pearson
// LHR: London Heathrow

更多關(guān)于for-in循環(huán)的信息缅阳,參見For 循環(huán)。

通過訪問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 的字典類型是無序集合類型。為了以特定的順序遍歷字典的鍵或值超棺,可以對字典的keys或values屬性使用sort()方法向族。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市棠绘,隨后出現(xiàn)的幾起案子件相,更是在濱河造成了極大的恐慌,老刑警劉巖氧苍,帶你破解...
    沈念sama閱讀 210,835評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件夜矗,死亡現(xiàn)場離奇詭異,居然都是意外死亡让虐,警方通過查閱死者的電腦和手機(jī)紊撕,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 89,900評論 2 383
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來赡突,“玉大人对扶,你說我怎么就攤上這事〔宴郑” “怎么了浪南?”我有些...
    開封第一講書人閱讀 156,481評論 0 345
  • 文/不壞的土叔 我叫張陵,是天一觀的道長漱受。 經(jīng)常有香客問我络凿,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,303評論 1 282
  • 正文 為了忘掉前任喷众,我火速辦了婚禮各谚,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘到千。我一直安慰自己昌渤,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,375評論 5 384
  • 文/花漫 我一把揭開白布憔四。 她就那樣靜靜地躺著膀息,像睡著了一般。 火紅的嫁衣襯著肌膚如雪了赵。 梳的紋絲不亂的頭發(fā)上潜支,一...
    開封第一講書人閱讀 49,729評論 1 289
  • 那天,我揣著相機(jī)與錄音柿汛,去河邊找鬼冗酿。 笑死,一個(gè)胖子當(dāng)著我的面吹牛络断,可吹牛的內(nèi)容都是我干的裁替。 我是一名探鬼主播,決...
    沈念sama閱讀 38,877評論 3 404
  • 文/蒼蘭香墨 我猛地睜開眼貌笨,長吁一口氣:“原來是場噩夢啊……” “哼弱判!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起锥惋,我...
    開封第一講書人閱讀 37,633評論 0 266
  • 序言:老撾萬榮一對情侶失蹤昌腰,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后膀跌,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體遭商,經(jīng)...
    沈念sama閱讀 44,088評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,443評論 2 326
  • 正文 我和宋清朗相戀三年淹父,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了株婴。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片怎虫。...
    茶點(diǎn)故事閱讀 38,563評論 1 339
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡暑认,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出大审,到底是詐尸還是另有隱情蘸际,我是刑警寧澤,帶...
    沈念sama閱讀 34,251評論 4 328
  • 正文 年R本政府宣布徒扶,位于F島的核電站粮彤,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜导坟,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,827評論 3 312
  • 文/蒙蒙 一屿良、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧惫周,春花似錦尘惧、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,712評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至登舞,卻和暖如春贰逾,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背菠秒。 一陣腳步聲響...
    開封第一講書人閱讀 31,943評論 1 264
  • 我被黑心中介騙來泰國打工疙剑, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人践叠。 一個(gè)月前我還...
    沈念sama閱讀 46,240評論 2 360
  • 正文 我出身青樓核芽,卻偏偏與公主長得像,于是被迫代替她去往敵國和親酵熙。 傳聞我的和親對象是個(gè)殘疾皇子轧简,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,435評論 2 348

推薦閱讀更多精彩內(nèi)容