为什么在执行performSelectorInBackground时没有自动释放池:?

时间:2009-05-30 10:16:37

标签: iphone multithreading cocoa-touch uikit autorelease

我正在调用后台线程中的方法:

[self performSelectorInBackground:@selector(loadViewControllerWithIndex:) withObject:[NSNumber numberWithInt:viewControllerIndex]];

然后,我有这个方法实现,由选择器调用:

- (void) loadViewControllerWithIndex:(NSNumber *)indexNumberObj {
    NSAutoreleasePool *arPool = [[NSAutoreleasePool alloc] init];
    NSInteger vcIndex = [indexNumberObj intValue];

    Class c;
    UIViewController *controller = [viewControllers objectAtIndex:vcIndex];

    switch (vcIndex) {
        case 0:
            c = [MyFirstViewController class];
            break;
        case 1:
            c = [MySecondViewController class];
            break;
        default:
            NSLog(@"unknown index for loading view controller: %d", vcIndex); // error
            break;
    }

    if ((NSNull *)controller == [NSNull null]) {
        controller = [[c alloc] initWithNib];
        [viewControllers replaceObjectAtIndex:vcIndex withObject:controller];
        [controller release];
    }

    if (controller.view.superview == nil) {
        UIView *placeholderView = [viewControllerPlaceholderViews objectAtIndex:vcIndex];
        [placeholderView addSubview:controller.view];
    }

    [arPool release];
}

Althoug我确实为那个线程创建了一个自动释放池,我总是得到这个错误:

2009-05-30 12:03:09.910 Demo[1827:3f03] *** _NSAutoreleaseNoPool(): Object 0x523e50 of class NSCFNumber autoreleased with no pool in place - just leaking
Stack: (0x95c83f0f 0x95b90442 0x28d3 0x2d42 0x95b96e0d 0x95b969b4 0x93a00155 0x93a00012)

如果我拿走自动释放池,我会得到一大堆这样的消息。我还尝试在performSelectorInBackground:的调用周围创建一个自动释放池,但这没有帮助。

我怀疑参数,但我不知道为什么编译器会抱怨NSCFNumber。我错过了什么吗?

我的实例变量都是“非原子的”。这可能是个问题吗?

更新:我可能还怀疑某个变量已被添加到主线程的自动释放池(可能是一个ivar),现在它试图在错误的自动释放池中释放那个?如果是这样,我该如何解决? (该死的,这种穿线的东西很复杂;))

2 个答案:

答案 0 :(得分:6)

最有可能的原因是因为泄漏的对象(NSNumber)是从线程外部传入的参数。因此,此变量属于调用线程(及其自动释放池)

线程调用周围的自动释放池不起作用的原因是因为线程创建者(performSelectorInbackground) - 立即返回,很可能是在线程仍在运行时。

我建议你在将参数作为参数传递之后对你的选择器参数进行一次发布。

答案 1 :(得分:2)

我同意最有可能的原因是因为泄漏的对象(NSNumber)是从线程外部传入的参数。因此,此变量属于调用线程(及其自动释放池)

调用线程应使用NSAutoreleasePool 我建议您在参数中添加一条保留指令:

- (void) loadViewControllerWithIndex:(NSNumber *)indexNumberObj {
    NSAutoreleasePool *arPool = [[NSAutoreleasePool alloc] init];
    [indexNumberObj retain];

    ....

    [arPool release];
  }