- 版權聲明:本文為博主原創文章,未經博主允許不得轉載。

該文簡短介紹在iOS開發中遍歷字典、數組和集合的幾種常見方式。
一、使用for循環
要遍歷字典、數組或者是集合,for循環是最簡單也用的比較多的方法,示例如下:
//普通的for循環遍歷
-(void)iteratorWithFor
{
//////////處理數組//////////
NSArray *arrayM = @[@"1",@"2",@"3",@"4"];
NSInteger arrayMCount = [arrayM count];
for (int i = 0; i<arrayMCount; i++) {
NSString *obj = arrayM[i];
NSLog(@"%@",obj);
}
//////////處理字典//////////
NSDictionary *dictM = @{@"1":@"one",@"2":@"two",@"3":@"three"};
NSArray *dictKeysArray = [dictM allKeys];
for (int i = 0; i<dictKeysArray.count; i++) {
NSString *key = dictKeysArray[i];
NSString *obj = [dictM objectForKey:key];
NSLog(@"%@:%@",key,obj);
}
//////////處理集合//////////
NSSet * setM = [[NSSet alloc] initWithObjects:@"one",@"two",@"three",@"four", nil];
NSArray *setObjArray = [setM allObjects];
for (int i = 0; i<setObjArray.count; i++) {
NSString *obj = setObjArray[i];
NSLog(@"%@",obj);
}
//////////反向遍歷----降序遍歷----以數組為例
NSArray *arrayM2 = @[@"1",@"2",@"3",@"4"];
NSInteger arrayMCount2 = [arrayM2 count] - 1;
for (NSInteger i = arrayMCount2; i>0; i--) {
NSString *obj = arrayM2[i];
NSLog(@"%@",obj);
}
}```
>優點:簡單
>缺點:由于字典和集合內部是無序的,導致我們在遍歷字典和集合的時候需要借助一個新的『數組』作為中介來處理,多出了一部分開銷。
###二、使用NSEnumerator遍歷
NSEnumerator的使用和基本的for循環類似,不過代碼量要大一些。示例如下:
//使用NSEnumerator遍歷
-(void)iteratorWithEnumerator
{
//////////處理數組//////////
NSArray *arrayM = @[@"1",@"2",@"3",@"4"];
NSEnumerator *arrayEnumerator = [arrayM objectEnumerator];
NSString *obj;
while ((obj = [arrayEnumerator nextObject]) != nil) {
NSLog(@"%@",obj);
}
//////////處理字典//////////
NSDictionary *dictM = @{@"1":@"one",@"2":@"two",@"3":@"three"};
NSEnumerator *dictEnumerator = [dictM keyEnumerator];
NSString *key;
while ((key = [dictEnumerator nextObject]) != nil) {
NSString *obj = dictM[key];
NSLog(@"%@",obj);
}
//////////處理集合//////////
NSSet * setM = [[NSSet alloc] initWithObjects:@"one",@"two",@"three",@"four", nil];
NSEnumerator *setEnumerator = [setM objectEnumerator];
NSString *setObj;
while ((setObj = [setEnumerator nextObject]) != nil) {
NSLog(@"%@",setObj);
}
//////////反向遍歷----降序遍歷----以數組為例
NSArray *arrayM2 = @[@"1",@"2",@"3",@"4"];
NSEnumerator *arrayEnumerator2 = [arrayM2 reverseObjectEnumerator];
NSString *obj2;
while ((obj2 = [arrayEnumerator2 nextObject]) != nil) {
NSLog(@"%@",obj2);
}
}```
優點:對于不同的數據類型,遍歷的語法相似;內部可以簡單的通過reverseObjectEnumerator設置進行反向遍歷。
缺點:代碼量稍大。
三、使用for...in遍歷
在Objective-C 2.0 中增加了for ...In 形式的快速遍歷。此種遍歷方式語法簡潔,速度飛快。示例如下:
//使用for...In進行快速遍歷
-(void)iteratorWithForIn
{
//////////處理數組//////////
NSArray *arrayM = @[@"1",@"2",@"3",@"4"];
for (id obj in arrayM) {
NSLog(@"%@",obj);
}
//////////處理字典//////////
NSDictionary *dictM = @{@"1":@"one",@"2":@"two",@"3":@"three"};
for (id obj in dictM) {
NSLog(@"%@",dictM[obj]);
}
//////////處理集合//////////
NSSet * setM = [[NSSet alloc] initWithObjects:@"one",@"two",@"three",@"four", nil];
for (id obj in setM) {
NSLog(@"%@",obj);
}
//////////反向遍歷----降序遍歷----以數組為例
NSArray *arrayM2 = @[@"1",@"2",@"3",@"4"];
for (id obj in [arrayM2 reverseObjectEnumerator]) {
NSLog(@"%@",obj);
}
}```
>優點:1)語法簡潔;2)效率最高;
>缺點:無法獲得當前遍歷操作所針對的下標。
###四、基于Block的遍歷方式
基于Block的方式來進行遍歷是最新引入的方法。它提供了遍歷數組|字典等類型數據的最佳實踐。示例如下:
//基于塊(block)的遍歷方式
-(void)iteratorWithBlock
{
//////////處理數組//////////
NSArray *arrayM = @[@"1",@"2",@"3",@"4"];
[arrayM enumerateObjectsUsingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
NSLog(@"%zd--%@",idx,obj);
}];
//////////處理字典//////////
NSDictionary *dictM = @{@"1":@"one",@"2":@"two",@"3":@"three"};
[dictM enumerateKeysAndObjectsUsingBlock:^(id _Nonnull key, id _Nonnull obj, BOOL * _Nonnull stop) {
NSLog(@"%@:%@",key,obj);
}];
//////////處理集合//////////
NSSet * setM = [[NSSet alloc] initWithObjects:@"one",@"two",@"three",@"four", nil];
[setM enumerateObjectsUsingBlock:^(id _Nonnull obj, BOOL * _Nonnull stop) {
NSLog(@"%@",obj);
}];
//////////反向遍歷----降序遍歷----以數組為例
NSArray *arrayM2 = @[@"1",@"2",@"3",@"4"];
[arrayM2 enumerateObjectsWithOptions:NSEnumerationReverse usingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
NSLog(@"%zd--%@",idx,obj);
}];
}```
優點:1)遍歷時可以直接從block中獲得需要的所有信息,包括下標、值等。特別相對于字典而言,不需要做多余的編碼即可同時獲得key和value的值。
2)能夠直接修改block中key或者obj的類型為真實類型,可以省去類型轉換的工作。
3)可以通過NSEnumerationConcurrent枚舉值開啟并發迭代功能。
說明:基于Block的遍歷方式在實現反向遍歷的時候也非常簡單,使用enumerateObjectsWithOptions方法,傳遞NSEnumerationReverse作為參數即可,在處理遍歷操作的時候推薦基于Block的遍歷方式。
五、使GCD中的dispatch_apply函數
使用GCD中的dispatch_apply函數也能實現字典、數組等的遍歷,該函數比較適合處理耗時較長、迭代次數較多的情況。示例如下:
//使用GCD中的dispatch_apply函數
-(void)iteratorWithApply
{
//////////處理數組//////////
NSArray *arrayM = @[@"1",@"2",@"3",@"4"];
//獲得全局并發隊列
dispatch_queue_t queue = dispatch_get_global_queue(0, 0);
dispatch_apply(arrayM.count, queue, ^(size_t index) {
NSLog(@"%@--%@",arrayM[index],[NSThread currentThread]);
});
}```
>優點:開啟多條線程并發處理遍歷任務,執行效率高。
>缺點:1)對于字典和集合的處理需借助數組;2)無法實現反向遍歷。