NSEnumerator 性能与 Cocoa 中的 for 循环

发布于 2024-07-04 16:20:16 字数 106 浏览 9 评论 0原文

我知道,如果您有一个修改循环中项目计数的循环,则在集合上使用 NSEnumerator 是确保代码崩溃的最佳方法,但是我想了解 NSEnumerator 类之间的性能权衡只是一个老派的 for 循环

I know that if you have a loop that modifies the count of the items in the loop, using the NSEnumerator on a set is the best way to make sure your code blows up, however I would like to understand the performance tradeoffs between the NSEnumerator class and just an old school for loop

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(3

听闻余生 2024-07-11 16:20:16

在 Objective-C 2.0 中使用新的 for (... in ...) 语法通常是迭代集合的最快方法,因为它可以在堆栈上维护一个缓冲区并获取批量的项目进去。

使用 NSEnumerator 通常是最慢的方法,因为它经常复制正在迭代的集合; 对于不可变集合,这可能很便宜(相当于 -retain),但对于可变集合,它可能会导致创建不可变副本。

进行您自己的迭代 - 例如,使用 -[NSArray objectAtIndex:] - 通常会介于两者之间,因为虽然您不会有潜在的复制开销,但您也不会获得批量的来自底层集合的对象。

(PS - 这个问题应该标记为 Objective-C,而不是 C,因为 NSEnumerator 是一个 Cocoa 类,并且新的 for (... in ...) 语法特定于 Objective-C。)

Using the new for (... in ...) syntax in Objective-C 2.0 is generally the fastest way to iterate over a collection because it can maintain a buffer on the stack and get batches of items into it.

Using NSEnumerator is generally the slowest way because it often copies the collection being iterated; for immutable collections this can be cheap (equivalent to -retain) but for mutable collections it can cause an immutable copy to be created.

Doing your own iteration — for example, using -[NSArray objectAtIndex:] — will generally fall somewhere in between because while you won't have the potential copying overhead, you also won't be getting batches of objects from the underlying collection.

(PS - This question should be tagged as Objective-C, not C, since NSEnumerator is a Cocoa class and the new for (... in ...) syntax is specific to Objective-C.)

浪漫之都 2024-07-11 16:20:16

运行测试几次后,结果几乎是一样的。 每个测量块连续运行 10 次。

在我的例子中,结果从最快到最慢:

  1. For..in (testPerformanceExample3) (0.006 sec)
  2. While (testPerformanceExample4) (0.026 秒)
  3. For(;;) (testPerformanceExample1) (0.027 秒)
  4. 枚举块 (testPerformanceExample2) (0.067 秒)

for 和 while 循环几乎相同。

迭代之间的比较

tmp 是一个 NSArray,包含 100 万个从 0 到 999999 的对象。

- (NSArray *)createArray
{
    self.tmpArray = [NSMutableArray array];
    for (int i = 0; i < 1000000; i++)
    {
        [self.tmpArray addObject:@(i)];
    }
    return self.tmpArray;
}

整个代码:

ViewController.h

#import <UIKit/UIKit.h>

@interface ViewController : UIViewController

@property (strong, nonatomic) NSMutableArray *tmpArray;
- (NSArray *)createArray;

@end

ViewController.m

#import "ViewController.h"

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    [self createArray];
}

- (NSArray *)createArray
{
    self.tmpArray = [NSMutableArray array];
    for (int i = 0; i < 1000000; i++)
    {
        [self.tmpArray addObject:@(i)];
    }
    return self.tmpArray;
}

@end

MyTestfile.m

#import <UIKit/UIKit.h>
#import <XCTest/XCTest.h>

#import "ViewController.h"

@interface TestCaseXcodeTests : XCTestCase
{
    ViewController *vc;
    NSArray *tmp;
}

@end

@implementation TestCaseXcodeTests

- (void)setUp {
    [super setUp];
    vc = [[ViewController alloc] init];
    tmp = vc.createArray;
}

- (void)testPerformanceExample1
{
    [self measureBlock:^{
        for (int i = 0; i < [tmp count]; i++)
        {
            [tmp objectAtIndex:i];
        }
    }];
}

- (void)testPerformanceExample2
{
    [self measureBlock:^{
        [tmp enumerateObjectsUsingBlock:^(NSNumber *obj, NSUInteger idx, BOOL *stop) {
           obj;
        }];
    }];
}

- (void)testPerformanceExample3
{
    [self measureBlock:^{
        for (NSNumber *num in tmp)
        {
            num;
        }
    }];
}

- (void)testPerformanceExample4
{
    [self measureBlock:^{
        int i = 0;
        while (i < [tmp count])
        {
            [tmp objectAtIndex:i];
            i++;
        }
    }];
}

@end

有关更多信息,请访问: Apple“关于使用 Xcode 进行测试”

After running the test several times, the result is almost the same. Each measure block runs 10 times consecutively.

The result in my case from the fastest to the slowest:

  1. For..in (testPerformanceExample3) (0.006 sec)
  2. While (testPerformanceExample4) (0.026 sec)
  3. For(;;) (testPerformanceExample1) (0.027 sec)
  4. Enumeration block (testPerformanceExample2) (0.067 sec)

The for and while loop is almost the same.

comparation between iterations

The tmp is an NSArray which contains 1 million objects from 0 to 999999.

- (NSArray *)createArray
{
    self.tmpArray = [NSMutableArray array];
    for (int i = 0; i < 1000000; i++)
    {
        [self.tmpArray addObject:@(i)];
    }
    return self.tmpArray;
}

The whole code:

ViewController.h

#import <UIKit/UIKit.h>

@interface ViewController : UIViewController

@property (strong, nonatomic) NSMutableArray *tmpArray;
- (NSArray *)createArray;

@end

ViewController.m

#import "ViewController.h"

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    [self createArray];
}

- (NSArray *)createArray
{
    self.tmpArray = [NSMutableArray array];
    for (int i = 0; i < 1000000; i++)
    {
        [self.tmpArray addObject:@(i)];
    }
    return self.tmpArray;
}

@end

MyTestfile.m

#import <UIKit/UIKit.h>
#import <XCTest/XCTest.h>

#import "ViewController.h"

@interface TestCaseXcodeTests : XCTestCase
{
    ViewController *vc;
    NSArray *tmp;
}

@end

@implementation TestCaseXcodeTests

- (void)setUp {
    [super setUp];
    vc = [[ViewController alloc] init];
    tmp = vc.createArray;
}

- (void)testPerformanceExample1
{
    [self measureBlock:^{
        for (int i = 0; i < [tmp count]; i++)
        {
            [tmp objectAtIndex:i];
        }
    }];
}

- (void)testPerformanceExample2
{
    [self measureBlock:^{
        [tmp enumerateObjectsUsingBlock:^(NSNumber *obj, NSUInteger idx, BOOL *stop) {
           obj;
        }];
    }];
}

- (void)testPerformanceExample3
{
    [self measureBlock:^{
        for (NSNumber *num in tmp)
        {
            num;
        }
    }];
}

- (void)testPerformanceExample4
{
    [self measureBlock:^{
        int i = 0;
        while (i < [tmp count])
        {
            [tmp objectAtIndex:i];
            i++;
        }
    }];
}

@end

For more information visit: Apples "About Testing with Xcode"

绮烟 2024-07-11 16:20:16

他们非常相似。 在 Objective-C 2.0 中,大多数枚举现在默认为 NSFastEnumeration,它创建一个集合中每个对象的地址缓冲区,然后可以传递该缓冲区。 与经典 for 循环相比,您节省的一个步骤是不必每次在循环内调用 objectAtIndex:i。 您正在枚举的集合的内部实现了快速枚举,而无需调用 objectAtIndex:i 方法

缓冲区是您在枚举时无法改变集合的部分原因,对象的地址将更改并且构建的缓冲区将不再匹配。

作为奖励,2.0 中的格式看起来和经典的 for 循环一样漂亮:

for ( Type newVariable in expression ) { 
    stmts 
}

阅读以下文档以深入了解:
NSFastEnumeration 协议参考

They are very similar. With Objective-C 2.0 most enumerations now default to NSFastEnumeration which creates a buffer of the addresses to each object in the collection that it can then deliver. The one step that you save over the classic for loop is not having to call objectAtIndex:i each time inside the loop. The internals of the collection you are enumerating implement fast enumeration with out calling objectAtIndex:i method.

The buffer is part of the reason that you can't mutate a collection as you enumerate, the address of the objects will change and the buffer that was built will no longer match.

As a bonus the format in 2.0 looks as nice as the classic for loop:

for ( Type newVariable in expression ) { 
    stmts 
}

Read the following documentaion to go deeper:
NSFastEnumeration Protocol Reference

~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文