10

我一遍又一遍地看到这一点,为什么在循环中使用快速枚举而不是NSEnumeratorusing更快nextObject:

4

3 回答 3

29

NSEnumerator是枚举集合的旧方法。它涉及创建一个对象来表示枚举,然后为每次迭代调用一个方法。虽然这多年来一直可以完美使用,但效率并不高,因为它涉及循环的每次迭代至少发送一条消息。NSFastEnumeration是更现代的方法,它利用本地语言支持来提供更有效的枚举。它在后台工作的方式是创建一个表示当前枚举状态的结构并重复调用-countByEnumeratingWithState:objects:count:集合。此方法在objectsout-param 中返回一个 C 对象数组以及在count参数外。这允许调用者然后迭代 C 数组。本质上,这意味着每个对象块调用一次消息,根据集合的不同,这可能与获取所有对象的单个消息调用一样有效。

如果你有一些看起来像的代码

for (id obj in myArray) {
    [obj doSomething];
}

这被编译器翻译成大致相当于

NSFastEnumerationState __enumState = {0};
id __objects[MAX_STACKBUFF_SIZE];
NSUInteger __count;
while ((__count = [myArray countByEnumeratingWithState:&__enumState objects:__objects count:MAX_STACKBUFF_SIZE]) > 0) {
    for (NSUInteger i = 0; i < __count; i++) {
        id obj = __objects[i];
        [obj doSomething];
    }
}

实际使用的变量是隐藏的,对象缓冲区的最大大小也取决于实现,但基本思想是有的。它将对 obj-c 集合的迭代转换为对 C 数组的迭代。

于 2011-03-02T04:18:20.520 回答
2

它与 Apple 的实现不同,但有助于理解。

- (NSUInteger) countByEnumeratingWithState: (NSFastEnumerationState*)state  
                   objects: (id*)stackbuf
                     count: (NSUInteger)len
{
  IMP nextObject = [self methodForSelector: @selector(nextObject)];
  int i;

  state->itemsPtr = stackbuf;
  state->mutationsPtr = (unsigned long*)self;
  for (i = 0; i < len; i++)
    {
      id next = nextObject(self, @selector(nextObject));

      if (nil == next)
    {
      return i;
    }
      *(stackbuf+i) = next;
    }
  return len;
}
于 2011-03-02T04:30:02.673 回答
0
NSArray *array = something;

数组 = { {1,2}, {2,3}, {3,4} }

这意味着数组是数组的数组。那么如何访问所有数组及其值。我们可以像这样使用 for 循环

for (int i = 0; i < array.count; i++)
{
    NSArray x = [array objectAtIndex:i];
}

或快速枚举像这样工作

for(NSArray array2 in array)
{
   // do what ever you want with this new array2.
}

这是一个示例。
PS。我忘记了数组在控制台中的外观。

于 2011-03-02T04:16:58.440 回答