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