IOS开发中遍历的优势

来源:爱站网时间:2021-02-02编辑:网友分享
遍历通常都是利用CPU的优势进行运行,而在IOS开发中也很多遍历的操作,今天爱站技术频道小编为大家整合了IOS开发中遍历的优势,一起跟随爱站技术频道小编看看吧。

遍历通常都是利用CPU的优势进行运行,而在IOS开发中也很多遍历的操作,今天爱站技术频道小编为大家整合了IOS开发中遍历的优势,一起跟随爱站技术频道小编看看吧。

经典for循环

NSArray *iosArray = @[@"a", @"b", @"c", @"d", @"e", @"f", @"g"];
for (int i = 0; i 

NSEnumerator遍历

NSArray *iosArray = @[@"a", @"b", @"c", @"d", @"e", @"f", @"g"];
NSEnumerator *enumerator = [iosArray objectEnumerator];//正向遍历
// NSEnumerator *enumerator = [iosArray reverseObjectEnumerator];//反向遍历

id object;

while ((object = [enumerator nextObject]) != nil) {
  //处理枚举器中的数据
  NSLog(@"%@", object);
}

for-in快速遍历

NSArray *iosArray = @[@"a", @"b", @"c", @"d", @"e", @"f", @"g"];
for (NSString *obj in iosArray) {
  //处理数组中的数据
  NSLog(@"%@", obj);
}

EnumeratorBlock遍历

NSArray *iosArray = @[@"a", @"b", @"c", @"d", @"e", @"f", @"g"];
[iosArray enumerateObjectsUsingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
  NSLog(@"%@", obj);
  if ([obj isEqualToString:@"e"]) {
    *stop = YES;  // 跳出遍历
  }
}];

另外,EnumeratorBlock还支持反向遍历,并发遍历,并发遍历可以使用多核的优化,充分利用系统的资源。

反向遍历

NSArray *iosArray = @[@"a", @"b", @"c", @"d", @"e", @"f", @"g"];
[iosArray enumerateObjectsWithOptions:NSEnumerationReverse usingBlock:^(NSString *obj, NSUInteger idx, BOOL * _Nonnull stop) {
  NSLog(@"%@", obj);
  if ([obj isEqualToString:@"e"]) {
    *stop = YES;
  }
}];

并发遍历

NSArray *iosArray = @[@"a", @"b", @"c", @"d", @"e", @"f", @"g"];
NSMutableArray *iosMutableArray = [NSMutableArray arrayWithArray:iosArray];
[iosMutableArray enumerateObjectsWithOptions:NSEnumerationConcurrent usingBlock:^(NSString *obj, NSUInteger idx, BOOL * _Nonnull stop) {
  obj = [NSString stringWithFormat:@"_%@", obj];
  [iosMutableArray replaceObjectAtIndex:idx withObject:obj];
  NSLog(@"%@", obj);

  if ([obj isEqualToString:@"_I"]) {
    *stop = YES;
  }
}];

dispatch_apply遍历

dispatch_apply类似于for循环,这里需要注意的是,dispatch_apple是同步调用,调用完毕返回结果,并且由于是GCD实现,所以可以使用并发队列或者是串行队列。代码如下:

dispatch_queue_t queue = dispatch_queue_create("queue", DISPATCH_QUEUE_CONCURRENT);
//  dispatch_queue_t queue = dispatch_queue_create("queue", DISPATCH_QUEUE_SERIAL); // 串行队列
dispatch_apply(array.count, queue, ^(size_t i) {
  Enumerate *enumerate = [array objectAtIndex:i];
  NSLog(@"number: %ld", enumerate.number);
});

遍历的注意事项

for循环中不要修改数组

遍历过程中是不能随便删除遍历的元素的,如果需要删除元素,可以先复制一份出来,比如如下的代码会有问题:

NSMutableArray *iosArray = @[@"a", @"b", @"c", @"d", @"e", @"f", @"g"];
for (NSString *obj in iosArray) {
  //处理数组中的数据
  if([@"e" isEqualTo:obj]) {
    [iosArray removeObject:obj];
  }
}

但是使用enumerateBlock可以在block内部做removeObject操作,原因应该是和Block的特性有关, 在Block中会保存变量的值,而不会随变量的值的改变而改变 。

遍历的速率

当数组容量很大的时候,如果只是进行数组遍历的话,使用for-in是最快速的,其次是并发遍历,这个很多人都以为enumerateBlock是最快的。

遍历实践tips

数组分组

在开发中,有时需要对数组进行某种情况的分组,比如,一个拥有很多消息模型的数组,我们需要根据消息的创建月份进行分组,那么可以使用下面的方法实现:

NSMutableSet *set=[NSMutableSet set];
NSArray *array = @[message1, message2, message3, message4, message5, message6, message7];
__block NSArray *tempDataArray = [NSArray arrayWithArray:array];
[tempDataArray enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL *stop) {
  [set addObject:obj.month];//利用set不重复的特性,得到有多少组,根据数组中消息的月份属性
}];
[set enumerateObjectsUsingBlock:^(id obj, BOOL *stop) {//遍历set数组
  NSPredicate *predicate = [NSPredicate predicateWithFormat:@"SELF.month = %@", obj];//创建谓词筛选器
  NSArray *group = [tempDataArray filteredArrayUsingPredicate:predicate];//用数组的过滤方法得到新的数组,在添加的最终的数组
}

倒序遍历

倒序遍历也很常见,可以使用上面的反向遍历来实现。

set排序

这个和Emunerate其实没有关系,但是也很实用,我们知道set是无序的,但是有时需要实现有顺序的set,可以使用下面来实现:

//由于set无序,现将set转换成nsarray
NSArray *sortDescriptor = @[[[NSSortDescriptor alloc] initWithKey:@"self" ascending:NO]];
NSArray *sortSetArray = [set sortedArrayUsingDescriptors:sortDescriptor];

其实原理是将set转化成array来实现的。

遍历在我们的日常开发中非常常见,但是根据应用场景,我们需要选择合适的遍历方法,希望能对有需要的学生有所帮助,感谢大家一直以来对js.aizhan.com的支持。

上一篇:iOS中如何使用MD5加密字符串

下一篇:IOS开发中uilabel行间距和自适应高度的操作

您可能感兴趣的文章

相关阅读

热门软件源码

最新软件源码下载