iOS開發實用技巧—Objective-C中的各類遍歷(迭代)方式git
說明:github
1)該文簡短介紹在iOS開發中遍歷字典、數組和集合的幾種常見方式。數組
2)該文對應的代碼能夠在下面的地址得到:https://github.com/HanGangAndHanMeimei/Code併發
1、使用for循環app
要遍歷字典、數組或者是集合,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; i<arrayMCount; i++) { 8 NSString *obj = arrayM[i]; 9 NSLog(@"%@",obj); 10 } 11 12 //////////處理字典////////// 13 NSDictionary *dictM = @{@"1":@"one",@"2":@"two",@"3":@"three"}; 14 NSArray *dictKeysArray = [dictM allKeys]; 15 for (int i = 0; i<dictKeysArray.count; i++) { 16 NSString *key = dictKeysArray[i]; 17 NSString *obj = [dictM objectForKey:key]; 18 NSLog(@"%@:%@",key,obj); 19 } 20 21 //////////處理集合////////// 22 NSSet * setM = [[NSSet alloc] initWithObjects:@"one",@"two",@"three",@"four", nil]; 23 NSArray *setObjArray = [setM allObjects]; 24 for (int i = 0; i<setObjArray.count; i++) { 25 NSString *obj = setObjArray[i]; 26 NSLog(@"%@",obj); 27 } 28 29 //////////反向遍歷----降序遍歷----以數組爲例 30 NSArray *arrayM2 = @[@"1",@"2",@"3",@"4"]; 31 NSInteger arrayMCount2 = [arrayM2 count] - 1; 32 33 for (NSInteger i = arrayMCount2; i>0; i--) { 34 NSString *obj = arrayM2[i]; 35 NSLog(@"%@",obj); 36 } 37 }
優勢:簡單編碼
缺點:因爲字典和集合內部是無序的,致使咱們在遍歷字典和集合的時候須要藉助一個新的『數組』做爲中介來處理,多出了一部分開銷。spa
2、使用NSEnumerator遍歷線程
NSEnumerator的使用和基本的for循環相似,不過代碼量要大一些。示例以下:code
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設置進行反向遍歷。
缺點:代碼量稍大。
3、使用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)效率最高;
缺點:沒法得到當前遍歷操做所針對的下標。
4、基於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的遍歷方式。
5、使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)沒法實現反向遍歷。