for循环
for循环入门常用,可以正序、逆序查询数据。
// for数组遍历
for (int i = 0; i < self.traverseArray.count; i++) {
NSLog(@"%@",self.traverseArray[I]);
}
// for字典遍历
NSArray *dictionaryArray = [self.traverseDictionary allKeys];
for (int i = 0 ; i < dictionaryArray.count; i++) {
NSLog(@"key = %@",dictionaryArray[I]);
}
for in 循环
for in 遍历,又称快速遍历,与for循环区别就是看不到循环次数及索引情况。数组是有序的,for循环过程中也是有序的,for in遍历过程中是根据数组中数据添加顺序而定的,for in 不可逆序遍历 !
for (NSString *str in self.traverseArray) {
NSLog(@"%@",str);
}
枚举器
枚举器是一种苹果官方推荐的更加面向对象的一种遍历方式,相比于for循环,它具有高度解耦、面向对象、使用方便等优势
// 正序遍历数组
[self.traverseArray enumerateObjectsUsingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
NSLog(@"正序%@",obj);
}];
// 逆序遍历数组
[self.traverseArray enumerateObjectsWithOptions:NSEnumerationReverse usingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
NSLog(@"逆序%@",obj);
}];
// 遍历字典
#warning 字典是无序的不存在正序逆序
[self.traverseDictionary enumerateKeysAndObjectsUsingBlock:^(id key, id value, BOOL *stop) {
NSLog(@"key:%@->value%@",key,value);
}];
dispatch_apply函数
GCD dispatch_apply函数是一个同步调用,block任务执行n次后才返回。该函数比较适合处理耗时较长、迭代次数较多的情况。
dispatch_queue_t queue = dispatch_get_global_queue(0, 0);
dispatch_apply(self.traverseArray.count, queue, ^(size_t insex) {
NSLog(@"%@",self.traverseArray[insex]);
});
ReactiveCocoa 遍历方法
ReactiveCocoa 中遍历主要是针对元组RACTuple,对于数组、字典的遍历都会包装成RACTuple进行处理。
使用方法先集成ReactiveCocoa 点这里
集成方法有所改变
cocopods引入时 pod ‘ReactiveObjC’
项目引入中 #import "ReactiveObjC/ReactiveObjC.h"
// 数组遍历
[self.traverseArray.rac_sequence.signal subscribeNext:^(id x) {
NSLog(@"%@",x);
}];
// 字典遍历 相当于元组数据
[self.traverseDictionary.rac_sequence.signal subscribeNext:^(id x) {
// 解包元组,会把元组的值,按顺序给参数里面的变量赋值
RACTupleUnpack(NSString *key,NSString*value) = x;
NSLog(@"key=%@ value=%@",key,value);
}];
以上几种遍历方式 在100、10000、100000次遍历所耗时长
感谢文章分享:iOS循环遍历 性能对比
附加:for循环删除数组中的元素crash问题
NSMutableArray *array = [NSMutableArray arrayWithObjects:@"2",@"3",@"4",@"9",@"4",@"12",@"22",@"4",@"4",@"5",@"6",@"1", nil];
for (NSString *str in array) {
if ([str isEqualToString:@"4"]) {
[array removeObject:str];
}
}
运行后肯定会crash,原因是数组越界。这是为什么?当我们正序快速遍历时,如果删除了一个,那么没有遍历到的元素位置都会往前移动一位,这样系统就无法确定接下来遍历是从删除位置开始呢,还是从删除位置下一位开始呢?这样就造成程序crash了。但是我们想要遍历删除时要怎么做?解决这个问题的方法有好几种。
1.采用for循环遍历
for (int i = 0; i < array.count; i++) {
NSString *str = array[i];
if ([str isEqualToString:@"4"]) {
[array removeObject:str];
}
}
2.定义一个副本,遍历副本找到想要删除的元素,然后在原数组中删除对应的元素。
NSMutableArray *copyArray = [NSMutableArray arrayWithArray:array];
for (NSString *str in copyArray) {
if ([str isEqualToString:@"4"]) {
[array removeObject:str];
}
}
3.对数组逆序遍历,查找对应元素后删除
NSEnumerator *enumerator = [array reverseObjectEnumerator];
for (NSString *str in enumerator) {
if ([str isEqualToString:@"4"]) {
[array removeObject:str];
}
}