Objective-C
經典for循環
優點:
?簡單
缺點:?
由于字典和集合內部是無序的,導致我們在遍歷字典和集合的時候需要借助一個新的『數組』作為中介來處理,多出了一部分開銷
_arr = @[@"1", @"2", @"3", @"4", @"5"];
_dic = @{@"one" : @1,
@"two" : @2,
@"three" : @3,
@"four" : @4,
@"five" : @5};
_set = [[NSSet alloc] initWithObjects:@"1", @"2", @"3", @"4", @"5", nil];
//////////數組//////////
for (NSInteger i = 0; i < _arr.count; i++) {
NSString *str = [_arr objectAtIndex:i];
NSLog(@"for 數組:%@", str);
}
//輸出: for 數組:1
//輸出: for 數組:2
//輸出: for 數組:3
//輸出: for 數組:4
//輸出: for 數組:5
//////////字典//////////
for (NSInteger i = 0; i < _dic.count; i++) {
NSString *key = [_dic.allKeys objectAtIndex:i];
NSString *value = [_dic.allValues objectAtIndex:i];
//NSString *value = [_dic objectForKey:key];
NSLog(@"for 字典:%@ %@", key, value);
}
//輸出: for 字典:one 1
//輸出: for 字典:five 5
//輸出: for 字典:three 3
//輸出: for 字典:two 2
//輸出: for 字典:four 4
//////////集合//////////
NSArray *setArr = [_set allObjects];
for (NSInteger i = 0; i < setArr.count; i++) {
NSString *str = [setArr objectAtIndex:i];
NSLog(@"for 集合:%@", str);
}
//輸出: for 集合:3
//輸出: for 集合:1
//輸出: for 集合:4
//輸出: for 集合:2
//輸出: for 集合:5
//////////反向遍歷//////////
for (NSInteger i = _arr.count; i > 0; i--) {
NSString *str = [_arr objectAtIndex:(i - 1)];
NSLog(@"for 倒序%@", str);
}
//輸出: for 倒序5
//輸出: for 倒序4
//輸出: for 倒序3
//輸出: for 倒序2
//輸出: for 倒序1
NSEnumerator迭代器
優點:
? 1)對于不同的數據類型,遍歷的語法相似
? 2)內部可以簡單的通過reverseObjectEnumerator設置進行反向遍歷
缺點:
代碼量稍大
_arr = @[@"1", @"2", @"3", @"4", @"5"];
_dic = @{@"one" : @1,
@"two" : @2,
@"three" : @3,
@"four" : @4,
@"five" : @5};
_set = [[NSSet alloc] initWithObjects:@"1", @"2", @"3", @"4", @"5", nil];
//////////數組//////////
NSEnumerator *arrEnumerator = [_arr objectEnumerator];
NSString *str;
while ((str = [arrEnumerator nextObject]) != nil) {
NSLog(@"NSEnumerator 數組:%@", str);
}
//輸出: NSEnumerator 數組:1
//輸出: NSEnumerator 數組:2
//輸出: NSEnumerator 數組:3
//輸出: NSEnumerator 數組:4
//輸出: NSEnumerator 數組:5
//////////字典//////////
NSEnumerator *dicEnumerator = [_dic keyEnumerator];
NSString *key;
while ((key = [dicEnumerator nextObject]) != nil) {
NSString *value = [_dic objectForKey:key];
NSLog(@"NSEnumerator 字典:%@ %@", key, value);
}
//輸出: NSEnumerator 字典:one 1
//輸出: NSEnumerator 字典:five 5
//輸出: NSEnumerator 字典:three 3
//輸出: NSEnumerator 字典:two 2
//輸出: NSEnumerator 字典:four 4
//////////集合//////////
NSEnumerator *setEnumerator = [_set objectEnumerator];
NSString *setStr;
while ((setStr = [setEnumerator nextObject]) != nil) {
NSLog(@"NSEnumerator 集合:%@", setStr);
}
//輸出: NSEnumerator 集合:3
//輸出: NSEnumerator 集合:1
//輸出: NSEnumerator 集合:4
//輸出: NSEnumerator 集合:2
//輸出: NSEnumerator 集合:5
//////////倒序//////////
NSEnumerator *arrReverseEnumerator = [_arr reverseObjectEnumerator];
NSString *str2;
while ((str2 = [arrReverseEnumerator nextObject]) != nil) {
NSLog(@"NSEnumerator 倒序:%@", str2);
}
//輸出: NSEnumerator 倒序:5
//輸出: NSEnumerator 倒序:4
//輸出: NSEnumerator 倒序:3
//輸出: NSEnumerator 倒序:2
//輸出: NSEnumerator 倒序:1
for..in
優點:
? 1)語法簡潔
? 2)效率最高
缺點:
無法獲得當前遍歷操作所針對的下標
_arr = @[@"1", @"2", @"3", @"4", @"5"];
_dic = @{@"one" : @1,
@"two" : @2,
@"three" : @3,
@"four" : @4,
@"five" : @5};
_set = [[NSSet alloc] initWithObjects:@"1", @"2", @"3", @"4", @"5", nil];
//////////數組//////////
for (NSString *str in _arr) {
NSLog(@"for..in 數組:%@", str);
}
//輸出: for..in 數組:1
//輸出: for..in 數組:2
//輸出: for..in 數組:3
//輸出: for..in 數組:4
//輸出: for..in 數組:5
//////////字典//////////
for (NSString *key in _dic) {
NSString *value = [_dic objectForKey:key];
NSLog(@"for..in 字典:%@ %@", key, value);
}
//輸出: for..in 字典:one 1
//輸出: for..in 字典:five 5
//輸出: for..in 字典:three 3
//輸出: for..in 字典:two 2
//輸出: for..in 字典:four 4
//////////集合//////////
for (NSString *str in _set) {
NSLog(@"for..in 集合:%@", str);
}
//輸出: for..in 集合:3
//輸出: for..in 集合:1
//輸出: for..in 集合:4
//輸出: for..in 集合:2
//輸出: for..in 集合:5
//////////倒序//////////
for (NSString *str in [_arr reverseObjectEnumerator]) {
NSLog(@"for..in 倒序%@", str);
}
//輸出: for..in 倒序5
//輸出: for..in 倒序4
//輸出: for..in 倒序3
//輸出: for..in 倒序2
//輸出: for..in 倒序1
Block塊的出現__簡潔干練的遍歷
優點:
1)遍歷時可以直接從block中獲得需要的所有信息,包括下標、值等。特別相對于字典而言,不需要做多余的編碼即可同時獲得key和value的值
?2)能夠直接修改block中key或者obj的類型為真實類型,可以省去類型轉換的工作
3)可以通過NSEnumerationConcurrent枚舉值開啟并發迭代功能
說明:
基于Block的遍歷方式在實現反向遍歷的時候也非常簡單,使用enumerateObjectsWithOptions方法,傳遞NSEnumerationReverse作為參數即可,在處理遍歷操作的時候推薦基于Block的遍歷方式
_arr = @[@"1", @"2", @"3", @"4", @"5"];
_dic = @{@"one" : @1,
@"two" : @2,
@"three" : @3,
@"four" : @4,
@"five" : @5};
_set = [[NSSet alloc] initWithObjects:@"1", @"2", @"3", @"4", @"5", nil];
//////////數組//////////
[_arr enumerateObjectsUsingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
NSLog(@"enumerateBlock 數組:%@ 下標:%lu", obj, (unsigned long)idx);
}];
//輸出: enumerateBlock 數組:1 下標:0
//輸出: enumerateBlock 數組:2 下標:1
//輸出: enumerateBlock 數組:3 下標:2
//輸出: enumerateBlock 數組:4 下標:3
//輸出: enumerateBlock 數組:5 下標:4
//////////字典//////////
[_dic enumerateKeysAndObjectsUsingBlock:^(id _Nonnull key, id _Nonnull obj, BOOL * _Nonnull stop) {
NSLog(@"enumerateBlock 字典:%@ %@", key, obj);
}];
//輸出: enumerateBlock 字典:one 1
//輸出: enumerateBlock 字典:five 5
//輸出: enumerateBlock 字典:three 3
//輸出: enumerateBlock 字典:two 2
//輸出: enumerateBlock 字典:four 4
//////////集合//////////
[_set enumerateObjectsUsingBlock:^(id _Nonnull obj, BOOL * _Nonnull stop) {
NSLog(@"enumerateBlock 集合:%@", obj);
}];
//輸出: enumerateBlock 集合:3
//輸出: enumerateBlock 集合:1
//輸出: enumerateBlock 集合:4
//輸出: enumerateBlock 集合:2
//輸出: enumerateBlock 集合:5
//////////倒序//////////
[_arr enumerateObjectsWithOptions:NSEnumerationReverse usingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
NSLog(@"enumerateBlock 倒序%@", obj);
}];
//輸出: enumerateBlock 倒序5
//輸出: enumerateBlock 倒序4
//輸出: enumerateBlock 倒序3
//輸出: enumerateBlock 倒序2
//輸出: enumerateBlock 倒序1
block塊 結束本次循環,結束循環
NSArray *names = @[@"a", @"b", @"c"];
[names enumerateObjectsUsingBlock:^(NSString * _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
// 執行完 block 中的代碼后,結束循環,并退出
if ([obj isEqualToString:@"a"]) {
*stop = YES;
}
// 結束本次循環
if ([obj isEqualToString:@"b"]) {
return;
}
// 結束循環,并退出
if ([obj isEqualToString:@"c"]) {
*stop = YES;
return;
}
}];