第一種方式:for循環(huán)
Objective-C是基于C語(yǔ)言的强重,自然可以使用for循環(huán)
遍歷數(shù)組:
NSArray *iosArray = @[@"L", @"O", @"V", @"E", @"I", @"O", @"S"];
for (int i = 0; i < iosArray.count; i++) {
//處理數(shù)組中數(shù)據(jù)
NSLog(@"%@", iosArray[i]);
}
遍歷數(shù)組很簡(jiǎn)單沒(méi)問(wèn)題鞠呈,下面遍歷字典
遍歷字典:
NSDictionary *dict = @{@"1":@"11", @"2":@"22", @"3":@"33"};
NSArray *keysArray = [dict allKeys];
for (int i = 0; i < keysArray.count; i++) {
//根據(jù)鍵值處理字典中的每一項(xiàng)
NSString *key = keysArray[i];
NSString *value = dict[key];
NSLog(@"%@", value);
}
我們知道字典和set是無(wú)序的至会,所以我們無(wú)法根據(jù)特定的整數(shù)下標(biāo)來(lái)直接訪問(wèn)其中的值慢哈,于是需要先獲取字典中的鍵或者set中的所有對(duì)象,這樣就可以在獲取到的有序數(shù)組上進(jìn)行遍歷了。然而創(chuàng)建數(shù)組是要額外的開(kāi)銷(xiāo)的囤萤,還會(huì)多創(chuàng)建出一個(gè)數(shù)組對(duì)象,他會(huì)保留collection中的所有對(duì)象喝滞,占用了內(nèi)存阁将。
總結(jié)優(yōu)缺點(diǎn):
優(yōu)點(diǎn):被廣泛使用,容易接受右遭,操作簡(jiǎn)單做盅;
缺點(diǎn):遍歷字典和set是比較繁瑣缤削,會(huì)占用比較多的系統(tǒng)資源。
第二種方式:NSEnumerator
NSEnumerator是一個(gè)抽象基類(lèi)吹榴,其中定義了2個(gè)方法亭敢,使其子類(lèi)實(shí)現(xiàn):
- (nullable ObjectType)nextObject;
@property (readonly, copy) NSArray*allObjects;
其中nextObject是關(guān)鍵方法,它返回枚舉里的下一個(gè)對(duì)象图筹。每次調(diào)用改方法其帅刀,其內(nèi)部結(jié)構(gòu)都會(huì)更新,使得下一次調(diào)用方法時(shí)能返回下一個(gè)對(duì)象远剩。等到枚舉中全部的對(duì)象都已經(jīng)返回之后扣溺,在調(diào)用就會(huì)返回nil,表示達(dá)到了枚舉的末端瓜晤。
Foundation框架中的collection都實(shí)現(xiàn)了這種遍歷方式锥余,例如:
NSArray *iosArray = @[@"L", @"O", @"V", @"E", @"I", @"O", @"S"];
NSEnumerator *enumerator = [iosArray objectEnumerator];//正向遍歷
NSEnumerator *enumerator = [iosArray reverseObjectEnumerator];//反向遍歷
id object;
while ((object = [enumerator nextObject]) != nil) {
//處理枚舉器中的數(shù)據(jù)
NSLog(@"%@", object);
}
字典和set實(shí)現(xiàn)的方式相似,不同的是字典中有key和value痢掠,要根據(jù)具體的key取出value驱犹。同時(shí)提供了正向遍歷和反向遍歷。
總結(jié)優(yōu)缺點(diǎn):
優(yōu)點(diǎn):代碼更易讀足画,不需要定義額外的數(shù)組雄驹;
缺點(diǎn):1、無(wú)法直接獲取遍歷操作的下標(biāo)淹辞,需要另外聲明變量記錄医舆;
2、需要自行創(chuàng)建NSEnumerator對(duì)象桑涎,稍顯麻煩彬向。
第三種方式:快速遍歷
Objective-C 2.0引入了快速遍歷這一功能兼贡,快速遍歷與NSEnumerator差不多攻冷,然而語(yǔ)法更為簡(jiǎn)潔,它為for循環(huán)開(kāi)設(shè)了in關(guān)鍵字遍希,簡(jiǎn)化了遍歷collection所需的語(yǔ)法等曼,例如遍歷數(shù)組:
NSArray *iosArray = @[@"L", @"O", @"V", @"E", @"I", @"O", @"S"];
for (NSString *obj in iosArray) {
//處理數(shù)組中的數(shù)據(jù)
NSLog(@"%@", obj);
}
遍歷字典也同樣簡(jiǎn)單:
NSDictionary *dict = @{@"1":@"11", @"2":@"22", @"3":@"33"};
for (NSString *key in dict) {
//處理字典的鍵值
NSString *value = dict[key];
NSLog(@"%@", value);
}
反向遍歷可以使用for (NSString *obj in [iosArray reverseObjectEnumerator])
總結(jié)優(yōu)缺點(diǎn):
優(yōu)點(diǎn):語(yǔ)法簡(jiǎn)潔,使用方便凿蒜,效率高禁谦;
缺點(diǎn):1、無(wú)法方便獲取當(dāng)前遍歷的下標(biāo)废封;
2州泊、無(wú)法在遍歷過(guò)程中修改被遍歷的collection,否則會(huì)導(dǎo)致崩潰漂洋。
第四種方式:基于塊的遍歷方式
這才是本文的重點(diǎn)遥皂,也是筆者極力推薦使用的方法力喷,蘋(píng)果封裝了如此高效、優(yōu)雅演训、易用的一套接口不用多么浪費(fèi)弟孟。
遍歷數(shù)組:
NSArray *iosArray = @[@"L", @"O", @"V", @"E", @"I", @"O", @"S"];
[iosArray enumerateObjectsUsingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
NSLog(@"%@", obj);
if ([obj isEqualToString:@"E"]) {
*stop = YES;
}
}];
參數(shù)說(shuō)明:obj表示數(shù)組中的元素,idx表示元素的下標(biāo)样悟,stop可以控制遍歷何時(shí)停止拂募,在需要停止時(shí)令stop = YES即可(不要忘記前面的*)。
這種方法清晰明了窟她,數(shù)組元素陈症,下標(biāo)都可直接獲取,就連何時(shí)停止都很容易實(shí)現(xiàn)震糖,break都可以退休了爬凑,遍歷字典也同樣簡(jiǎn)單。
遍歷字典:
NSDictionary *dict = @{@"1":@"11", @"2":@"22", @"3":@"33"};
[dict enumerateKeysAndObjectsUsingBlock:^(id _Nonnull key, id _Nonnull obj, BOOL * _Nonnull stop) {
NSLog(@"%@", obj);
if ([obj isEqualToString:@"22"]) {
*stop = YES;
}
}];
你沒(méi)有看錯(cuò)试伙,就是這么簡(jiǎn)單嘁信,block直接把字典的key和value都給我們了,再也不用書(shū)寫(xiě)直白而繁瑣的代碼了疏叨。
注意:
若已知collection里對(duì)象的數(shù)據(jù)類(lèi)型潘靖,可以修改塊簽名,知道對(duì)象的精確類(lèi)型后蚤蔓,編譯器就可以檢測(cè)開(kāi)發(fā)者是否調(diào)用了該對(duì)象所不具有的方法卦溢,并在發(fā)現(xiàn)問(wèn)題時(shí)報(bào)錯(cuò)。
NSDictionary *dict = @{@"1":@"11", @"2":@"22", @"3":@"33"};
[dict enumerateKeysAndObjectsUsingBlock:^(NSString *key, NSString *obj, BOOL * _Nonnull stop) {
NSLog(@"%@", obj);
if ([obj isEqualToString:@"22"]) {
*stop = YES;
}
}];
如代碼秀又,直接把key和value的類(lèi)型修改成NSString類(lèi)型单寂。
反向遍歷:
反向遍歷也同樣方便,調(diào)用另外一個(gè)方法即可:
NSArray *iosArray = @[@"L", @"O", @"V", @"E", @"I", @"O", @"S"];
[iosArray enumerateObjectsWithOptions:NSEnumerationReverse usingBlock:^(NSString *obj, NSUInteger idx, BOOL * _Nonnull stop) {
NSLog(@"%@", obj);
if ([obj isEqualToString:@"E"]) {
*stop = YES;
}
}];
這個(gè)方法相對(duì)于正向遍歷多了一個(gè)枚舉類(lèi)型的參數(shù)NSEnumerationReverse吐辙,打開(kāi)這個(gè)選項(xiàng)就可以反向遍歷了宣决。
并發(fā)遍歷:
順著這個(gè)枚舉類(lèi)型的參數(shù),就會(huì)引出塊枚舉的另一大優(yōu)勢(shì):并發(fā)遍歷昏苏,參數(shù)是:NSEnumerationConcurrent尊沸,也就是可以同時(shí)遍歷collection中的幾個(gè)元素,具體數(shù)量根據(jù)系統(tǒng)資源而定贤惯。這樣會(huì)充分利用系統(tǒng)資源洼专,高效快捷的完成collection的遍歷,系統(tǒng)底層會(huì)通過(guò)GCD來(lái)處理并發(fā)事宜孵构,開(kāi)發(fā)者不需要擔(dān)心內(nèi)存和線程屁商,其他方式若要實(shí)現(xiàn)高效的并發(fā)遍歷十分有難度。通過(guò)塊枚舉遍歷颈墅,改變collection并不會(huì)引起崩潰蜡镶,代碼如下:
NSArray *iosArray = @[@"L", @"O", @"V", @"E", @"I", @"O", @"S"];
NSMutableArray *iosMutableArray = [NSMutableArray arrayWithArray:iosArray];
[iosMutableArray enumerateObjectsWithOptions:NSEnumerationConcurrent usingBlock:^(NSString *obj, NSUInteger idx, BOOL * _Nonnull stop) {
obj = [NSString stringWithFormat:@"_%@", obj];
[iosMutableArray replaceObjectAtIndex:idx withObject:obj];
NSLog(@"%@", obj);
if ([obj isEqualToString:@"_I"]) {
*stop = YES;
}
}];
優(yōu)缺點(diǎn)總結(jié):
優(yōu)點(diǎn):
1溯职、可以完美實(shí)現(xiàn)for循環(huán)的所有功能;
2帽哑、可以方便獲取集合中的每一項(xiàng)元素谜酒;
3、提供了循環(huán)遍歷的參數(shù)妻枕,NSEnumerationReverse用來(lái)實(shí)現(xiàn)倒序循環(huán)僻族。NSEnumerationConcurrent用來(lái)實(shí)現(xiàn)并發(fā)遍歷,兩個(gè)參數(shù)可以同時(shí)使用屡谐;
4述么、這種循環(huán)方式效率高,能夠提升程序性能愕掏,開(kāi)發(fā)者可以專注于業(yè)務(wù)邏輯度秘,而不必?fù)?dān)心內(nèi)存和線程的問(wèn)題;
5饵撑、當(dāng)開(kāi)啟NSEnumerationConcurrent選項(xiàng)時(shí)剑梳,可以實(shí)現(xiàn)for循環(huán)和快速遍歷無(wú)法輕易實(shí)現(xiàn)的并發(fā)循環(huán)功能,系統(tǒng)底層會(huì)通過(guò)GCD處理并發(fā)事宜滑潘,這樣可以充分利用系統(tǒng)和硬件資源垢乙,達(dá)到最優(yōu)的遍歷效果;
6语卤、可以修改塊簽名追逮,當(dāng)我們已經(jīng)明確集合中的元素類(lèi)型時(shí),可以把默認(rèn)的簽名id類(lèi)型修改成已知類(lèi)型粹舵,比如常見(jiàn)的NSString钮孵,這樣既可以節(jié)省系統(tǒng)資源開(kāi)銷(xiāo),也可以防止誤向?qū)ο蟀l(fā)送不存在的方法是引起的崩潰眼滤。
缺點(diǎn):
1巴席、很多開(kāi)發(fā)者不知道這種遍歷方式;
2柠偶、這里使用了block情妖,需要注意在block里容易引起的保留環(huán)問(wèn)題,比如使用self調(diào)用方法時(shí)诱担,把self轉(zhuǎn)化成若引用即可打破保留環(huán)。如:__weak __typeof(self)weakSelf = self 或者 __weak MyController *weakSelf = self; 在block里使用weakSelf即可电爹。
注意:
使用基于塊的遍歷時(shí)是可以修改遍歷的元素的蔫仙,不會(huì)導(dǎo)致崩潰,但是如果要?jiǎng)h除遍歷的元素會(huì)導(dǎo)致后面的元素?zé)o法遍歷而崩潰丐箩,解決辦法有2種摇邦,1恤煞、一種是復(fù)制一份原集合的副本,對(duì)副本進(jìn)行操作施籍,找出所要操作的元素后再處理原集合居扒;2、使用反向遍歷丑慎,反向遍歷刪除元素后不會(huì)導(dǎo)致崩潰喜喂。