【问题标题】:Cocoa-Touch: performSelectorOnMainThread: weird behavior + crashCocoa-Touch:performSelectorOnMainThread:奇怪的行为 + 崩溃
【发布时间】:2009-11-09 09:38:29
【问题描述】:

我有一种情况,我懒惰地从 www 加载图像。
这是一个项目列表,当一个项目被点击时,一个详细视图被推送到导航控制器。

在该详细视图中,该项目有一个图像,首先是默认图像,我想开始从 URL 加载它的图像。

所以我要做的是创建一个对象,该对象一旦初始化就会分离一个新线程,该线程依次加载内容,然后通知我的视图数据已加载:

// in MyLoader:
- (MyLoader *)initWithUrl:(NSURL *)url requester:(id)requester {
    self.url = url;
    self.requester = requester; // both are nonatomic, retain properties
    [self performSelectorInBackground:@selector(loadIt) withObject:nil];
}

- (void)loadIt {
    NSAutoreleasePool *arp = [[NSAutoreleasePool alloc] init];
    NSData *data = [NSData dataWithContentsOfURL:url];
    [requester performSelectorOnMainThread:@selector(dataReady) withObject:data waitUntilDone:YES;
    [arp release];
}

// in MyRequester:
- (void)somewhere {
    MyLoader *loader = [[[MyLoader] alloc] initWithUrl:someUrl requester:self] autorelease];
    // then I retain loader somewhere, it's more complicated but I have verified that it's properly retained.
}

几点说明:

  1. 首先我认为某些变量可能存在问题。我在performSelectorOnMainThread 之前放了一个断点,并确认datarequester 都可以。

  2. 然后我以为是跨线程传递NSData引起的,所以我改了withObject:nil。它仍然崩溃。

  3. 当我进一步调查时,崩溃很奇怪。我指定了waitUntilDone:YES,我在requesterdataReady 中放置了一个断点。但是performSelectorOnMainThread 调用返回(它到达它之后的断点),而没有到达dataReady 内部的断点。顺便说一句,- (void)dataReady:(NSData*) 的正文现在只包含int x = 1;(放置断点)。另外,我尝试设置waitUntilDone:NO,它仍然崩溃。

  4. 未执行选择器(未到达断点),但在调用后不久发生崩溃。

有人知道出了什么问题吗?

这很明显,但要明确一点,如果我只是注释掉 [requester performSelectorOnMainThread... 部分,它不会崩溃。

另外,这里有一个堆栈跟踪,但它一点用都没有。

#0  0x00a71004 in ___TERMINATING_DUE_TO_UNCAUGHT_EXCEPTION___ ()
#1  0x93436e3b in objc_exception_throw ()
#2  0x0028aca6 in __NSThreadPerformPerform ()
#3  0x00a098e1 in CFRunLoopRunSpecific ()
#4  0x00a08c48 in CFRunLoopRunInMode ()
#5  0x0005a78d in GSEventRunModal ()
#6  0x0005a852 in GSEventRun ()
#7  0x0168a003 in UIApplicationMain ()
#8  0x000028d4 in main (argc=1, argv=0xbffff100) at /Users/myName/Document/appName/main.m:14

【问题讨论】:

  • 我怀疑有更简单的方法来查看未捕获的异常,但尝试包装 int retVal = UIApplicationMain(argc, argv, nil, nil);在带有 try/catch 块的 main.m 文件中检查或打印出异常。或许能提供一些线索。我知道在这样的事情之后我可以继续(在调试器中),它最终会打印出原因。但这可能是由于我的 .gdbinit 文件中的一些设置。我的 .gdbinit 对 NSException 有一些断点...

标签: iphone objective-c cocoa-touch


【解决方案1】:

你有:

[requester performSelectorOnMainThread:@selector(dataReady) withObject:data waitUntilDone:YES;

应该是:

[requester performSelectorOnMainThread:@selector(dataReady:) withObject:data waitUntilDone:YES;

注意:@selector(dataReady:)(带冒号) 由于您将参数传递给该方法,因此假定数据就绪的定义如下:

- (void) dataReady:(NSData *)theData ...

【讨论】:

  • 缺少冒号,而不是分号。冒号是方法名的一部分。
  • 将来,您可以通过转到您的构建设置并在其他警告标志中添加 -Wundeclared-selector 来捕获此问题。这将导致在上述情况下抛出警告,其中选择器与任何已知方法都不匹配。
猜你喜欢
  • 1970-01-01
  • 2013-11-16
  • 2019-03-21
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2022-11-01
  • 2023-02-02
相关资源
最近更新 更多