Swift循環(huán)遍歷集合的方法總結(jié)分享
前言
之前分享總結(jié)過(guò)OC循環(huán)遍歷,文章點(diǎn)擊這里:iOS遍歷集合(NSArray,NSDictionary、NSSet)方法總結(jié)。隨著Swift的逐漸完善,自己使用Swift開(kāi)發(fā)的項(xiàng)目經(jīng)驗(yàn)和知識(shí)逐漸積累,是時(shí)候總結(jié)一下Swift的循環(huán)遍歷了。相信Swift一定會(huì)給你一些不一樣的東西,甚至是驚喜,感興趣的朋友們下面來(lái)一起看看吧。

第一種方式:for-in循環(huán)
OC延續(xù)了C語(yǔ)言的for循環(huán),在Swift中被徹底改造,我們無(wú)法再使用傳統(tǒng)形式的for循環(huán)了
遍歷數(shù)組和字典:
//遍歷數(shù)組
let iosArray = ["L", "O", "V", "E", "I", "O", "S"]
for index in 0...6 {
print(iosArray[index])
}
for index in 0..<6 {
print(iosArray[index])
}
for element in iosArray {
print(element)
}
//遍歷字典
let iosDict = ["1": "one", "2": "two", "3": "three", "4": "four"]
for (key, value) in iosDict {
print("\(key): \(value)")
}
//單獨(dú)遍歷字典的key和value
let keys = iosDict.keys
for k in keys {
print(k)
}
let values = iosDict.values
for v in values {
print(v)
}
如上遍歷數(shù)組使用了2種方式
1、第一種方式是Swift中普通的for循環(huán)語(yǔ)法,在索引index和遍歷范圍0...6之間用關(guān)鍵字in,這里要注意0...6的表示的范圍是:0<= index <= 6,而0..<6表示的是:0<= index < 6,這里要注意的是沒(méi)有:0<..6的形式。只要熟悉了Swift語(yǔ)法,以上這些并不難理解。
拓展1:0...6的形式還可以取出制定范圍的數(shù)組中的元素,代碼如下:
let sectionArray = iosArray[1...4] print(sectionArray) 輸出: ▿ 4 elements - 0 : "O" - 1 : "V" - 2 : "E" - 3 : "I"
拓展2:0...6的形式還可以用來(lái)初始化創(chuàng)建數(shù)組,代碼如下:
let numbers = Array(1...7) print(numbers) 輸出: ▿ 7 elements - 0 : 1 - 1 : 2 - 2 : 3 - 3 : 4 - 4 : 5 - 5 : 6 - 6 : 7
也就是說(shuō)以后遇到涉及范圍的情況都可以嘗試0...6這種形式,看看是否可以迅速獲取指定范圍內(nèi)的元素,可用的地方還有很多,小伙伴自己掘金吧。
2、第二種方式類似于OC中的快速遍歷,不需要索引直接就可以訪問(wèn)到數(shù)組中的元素,也很好理解。
字典的遍歷可分為同時(shí)或者分別遍歷key和value
1、同時(shí)遍歷key和value時(shí)利用了Swift的元組,元組可以把不同類型的值組合成一個(gè)復(fù)合的值,使用起來(lái)非常方便,這樣就可以同時(shí)拿到字典的key和value了。
2、單獨(dú)遍歷字典的key個(gè)value時(shí),需要注意的是,keys和values并不是Array,因此無(wú)法直接使用keys[0]的形式訪問(wèn),他們實(shí)際的類型是LazyMapCollection<[Key : Value], Key> ,顯然不是一個(gè)數(shù)組。
當(dāng)然我們可以將他們轉(zhuǎn)換成數(shù)組,如下:
//將字典的kyes轉(zhuǎn)換成數(shù)組 let keys = Array(iosDict.keys) print(keys[0])
由于字典是無(wú)序的,所有這么做的意義并不大。
第二種方式:Swift為for循環(huán)帶來(lái)的驚喜
將以下內(nèi)容單拿出來(lái)作為第二種方式不太合適,其實(shí)這部分還是屬于Swift的for-in循環(huán),單獨(dú)拿出來(lái)是出于對(duì)這種方式的喜愛(ài),也讓大家在看的時(shí)候更加醒目。
反向遍歷
//倒序遍歷數(shù)組
for index in (0...6).reversed() {
print(iosArray[index])
}
for element in iosArray.reversed() {
print(element)
}
//倒序遍歷字典
for (key, value) in iosDict.reversed() {
print("\(key): \(value)")
}
1、如上無(wú)論是0...6這種索引方式還是快速遍歷,都可直接調(diào)用reversed()函數(shù)輕松實(shí)現(xiàn)反向遍歷。
2、對(duì)于字典的反向遍歷,有些小伙伴可能會(huì)有些疑問(wèn),字典是無(wú)序的,反向和正向遍歷有區(qū)別嗎,似乎意義不大。這里需要說(shuō)明的是,字典的無(wú)序是說(shuō)不保證順序,但是在內(nèi)存中是按照順序排列的,只是這種順序不一定按照我們存入或者編碼的順序排列,因此字典的反向遍歷也是有意義的。
3、看過(guò)我去年總結(jié)的OC循環(huán)遍歷的小伙伴一定還記得,當(dāng)我們需要在遍歷集合時(shí)改變集合中的元素時(shí),正向遍歷會(huì)偶爾出現(xiàn)崩潰的問(wèn)題,尤其是數(shù)據(jù)量較大時(shí)幾乎每次都會(huì)崩潰,當(dāng)我們使用反向遍歷時(shí)就沒(méi)有崩潰的問(wèn)題了,在Swift中為了保證程序的穩(wěn)定,也建議在遍歷集合需要修改集合元素時(shí)采用反向遍歷。
拓展:reversed()函數(shù)實(shí)際上是返回給我們一個(gè)順序完全顛倒的集合,那么我們就可以利用這個(gè)函數(shù)得到一個(gè)倒序的集合,非常方便,代碼如下:
//獲取倒序數(shù)組 let reversedArray = Array(iosArray.reversed()) print(reversedArray)
forEach遍歷
如果還有小伙伴認(rèn)為for-in遍歷繁瑣,Swift還提供了一種更加簡(jiǎn)潔的遍歷方式forEach,代碼如下:
//使用forEach正向遍歷
iosArray.forEach { (word) in
print(word)
}
//使用forEach的反向遍歷
iosArray.reversed().forEach { (word) in
print(word)
}
注意:
1、不能使用“break”或者“continue”退出遍歷;
2、使用“return”結(jié)束當(dāng)前循環(huán)遍歷,這種方式只是結(jié)束了當(dāng)前閉包內(nèi)的循環(huán)遍歷,并不會(huì)跳過(guò)后續(xù)代碼的調(diào)用。
stride遍歷
stride遍歷分為
stride<T : Strideable>(from start: T, to end: T, by stride: T.Stride)
和
stride<T : Strideable>(from start: T, through end: T, by stride: T.Stride)
兩種遍歷方式,代碼如下:
//stride正向遍歷
for index in stride(from: 1, to: 6, by: 1) {
print(index)
print(iosArray[index])
}
//stride正向跳躍遍歷
for index in stride(from: 0, to: 6, by: 2) {
print(index)
print(iosArray[index])
}
//stride反向遍歷
for index in stride(from: 6, to: 1, by: -1) {
print(index)
print(iosArray[index])
}
//stride through正向遍歷
for index in stride(from: 0, through: 6, by: 1) {
print(index)
print(iosArray[index])
}
1、正如stride單詞的含義“大步跨過(guò)”,使用這種方式遍歷的好處自然是可以靈活的根據(jù)自己的需求遍歷,比如我們有時(shí)需要遍歷索引為偶數(shù)或者基數(shù)的元素,或者每隔3個(gè)元素遍歷一次等等類似的需求都可以輕松實(shí)現(xiàn);
2、stride遍歷同樣可以實(shí)現(xiàn)正向和反向的遍歷,在by后面添加正數(shù)表示遞增的正向遍歷,添加負(fù)數(shù)表示遞減的反向遍歷;
3、to和through兩種遍歷方式的不同在于to不包含后面的索引,而through包含后面的索引,以to: 6和through: 6為例,to:<6或者>6,through:<=6或者>=6,至于是<還是>取決于是正向遍歷還是反向遍歷。
第三種方式:基于塊的遍歷
OC擁有一套很優(yōu)雅基于快的遍歷,Swift保持了這套優(yōu)秀的接口,下面來(lái)看看Swift是如何使用的。
正向遍歷
//遍歷數(shù)組
for (n, c) in iosArray.enumerated() {
print("\(n): \(c)")
}
//遍歷字典
for (n, c) in iosDict.enumerated() {
print("\(n): \(c)")
}
注意:
1、(n, c)中n表示元素的輸入順序,c表示集合中的每一個(gè)元素;
2、由于數(shù)組是有序的,所以在數(shù)組中n自然也可以表示每一個(gè)元素在數(shù)組中索引,而字典是無(wú)序的,但是n依然會(huì)按照0、1、2...的順序輸入,因此不可以代表在字典中的索引。
反向遍歷
//反向遍歷數(shù)組
for (n, c) in iosArray.enumerated().reversed() {
print("\(n): \(c)")
}
//反向遍歷字典
for (n, c) in iosDict.enumerated().reversed() {
print("\(n): \(c)")
}
反向遍歷就是直接在enumerated()函數(shù)后調(diào)用reversed()函數(shù)。
總結(jié)
在總結(jié)OC循環(huán)遍歷時(shí),筆者極力推崇基于塊的循環(huán)遍歷,因?yàn)橄啾容^其他的遍歷方式,基于塊的循環(huán)遍歷實(shí)在是集優(yōu)雅與實(shí)用于一體的尤物,但是在Swift中情況發(fā)生了一些變化。
1、以上列舉了3大種遍歷方式,其實(shí)這種分類方式并不嚴(yán)謹(jǐn),他們只是展示的形式不一樣,本質(zhì)上都屬于for-in循環(huán)的變種,都是基于枚舉的循環(huán)遍歷,所以大家不必太較真他們的本質(zhì)區(qū)別,在形式上區(qū)分開(kāi)就可以,不影響我們的使用;
2、在OC中除了少數(shù)情況我們需要使用for (int i = 0; i < n; i++) 的方式,我們都推薦使用基于枚舉的快速遍歷。在Swift中舍棄了for (int i = 0; i < n; i++)的形式,帶給我們for index in 0...6,這并不只是語(yǔ)法格式的變化,從本質(zhì)上已經(jīng)完全不一樣,使用起來(lái)更加方便,也擁有更多的接口提供便利的功能。并且相比較其他的遍歷方式也有很多優(yōu)點(diǎn),因此在Swift中我們無(wú)法一邊倒的選擇一種方式,應(yīng)該根據(jù)情況選擇合適的方法。
3、在OC中基于塊的遍歷還有一種情況是并發(fā)遍歷,我在Swift中沒(méi)有找到相應(yīng)的方法,看了幾千行代碼也沒(méi)有發(fā)現(xiàn)蹤跡,有找到的小伙伴還請(qǐng)告知,感激不盡。當(dāng)然并發(fā)遍歷用的并不多,我們很多時(shí)候我們都希望集合的元素按序出現(xiàn),在時(shí)間和效率上也沒(méi)有區(qū)別。
結(jié)束語(yǔ)
拋開(kāi)我們的使用習(xí)慣,Swift在很多時(shí)候都要比OC甚至其他編程語(yǔ)言更加簡(jiǎn)潔實(shí)用,雖然現(xiàn)在Swift編程很多還是依賴于OC,甚至Swift這種編譯時(shí)的語(yǔ)言的動(dòng)態(tài)性依然是基于OC的運(yùn)行時(shí),還有大量的類似于UIKit的庫(kù)也都是基于OC。Swift集其他語(yǔ)言優(yōu)點(diǎn)于一身又不失個(gè)性,從for循環(huán)等一系列小的地方的改變就可以看出其獨(dú)特之處。完全取代OC還需時(shí)日,可此時(shí)投身Swift確是最好時(shí)機(jī),你還在等什么?
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作能帶來(lái)一定的幫助,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)腳本之家的支持。
相關(guān)文章
Swift高階函數(shù)contains?allSatisfy?reversed?lexicographicallyPr
這篇文章主要為大家介紹了Swift高階函數(shù)contains?allSatisfy?reversed?lexicographicallyPrecedes用法示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-06-06
Swift算法之二叉樹(shù)實(shí)現(xiàn)的方法示例
二叉樹(shù)是計(jì)算機(jī)科學(xué)中最基本也是最重要的樹(shù)型結(jié)構(gòu),最常見(jiàn)的二叉樹(shù)生成算法通常是使用遞歸或者其他描述類語(yǔ)言的方法來(lái)實(shí)現(xiàn)。本文主要介紹了Swift算法之二叉樹(shù)實(shí)現(xiàn)的方法,文中介紹的非常詳細(xì),對(duì)大家具有一定的參考價(jià)值,需要的朋友們下面來(lái)一起看看吧。2017-03-03
Swift中static和class關(guān)鍵字的深入講解
這篇文章主要給大家介紹了關(guān)于Swift中static和class關(guān)鍵字的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者使用Java具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-03-03
iOS開(kāi)發(fā)中Swift 指紋驗(yàn)證功能模塊實(shí)例代碼
本文給大家分享ios調(diào)用touchid代碼塊,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友參考下把2017-03-03

