禁用Zombies会导致EXC_BAD_ACCESS

时间:2014-06-12 22:35:59

标签: ios objective-c core-data instruments nszombieenabled

我在应用程序中遇到以下堆栈跟踪崩溃 -

Thread : Crashed: com.apple.main-thread
0  libobjc.A.dylib                0x39dfa66a objc_release + 9
1  libobjc.A.dylib                0x39dfb0d7 (anonymous namespace)::AutoreleasePoolPage::pop(void*) + 358
2  CoreFoundation                 0x2f4a6c69 _CFAutoreleasePoolPop + 16
3  CoreFoundation                 0x2f53c1cb __CFRunLoopRun + 1306
4  CoreFoundation                 0x2f4a6f0f CFRunLoopRunSpecific + 522
5  CoreFoundation                 0x2f4a6cf3 CFRunLoopRunInMode + 106
6  GraphicsServices               0x343ff663 GSEventRunModal + 138
7  UIKit                          0x31df216d UIApplicationMain + 1136
8  Batted                         0x0009db07 main (main.m:16)

当Scheme的诊断选项中未启用僵尸时,会发生崩溃。但是,当我启用它时,崩溃不会发生。

我已经阅读了其他一些有关此问题的问答,并且他们似乎都建议一旦看到这种行为,启用僵尸并在模拟器上运行Zombie Profile Instruments。

我试过了,但是仪器似乎并没有表明任何错误,应用程序正常运行。 有什么其他方法可以在这里做根源 - 导致这个问题? 我在模拟器中使用XCode 5.1和iOS 7.1。

更新1

找到导致问题的有问题的代码,但我仍然不确定它导致问题的原因。 我正在使用CoreData,在NSManagedObject的子类中,我有 -

- (void)willTurnIntoFault;
{
    [super willTurnIntoFault];

    if ([self observationInfo])
    {
        BNLogInfo(@"%@ has observers:\n%@", [self objectID], [self observationInfo]);
    }
}

在上面的代码中,[self observationInfo]是违规行。

当应用程序启动时,我在枚举块中循环遍历一些NSManagedObjects并在其上设置一些属性,这反过来触发willTurnIntoFault方法。一旦枚举块完成,就会发生崩溃。

神秘之处在于崩溃并不会在这个方法中发生,但如果没有这个方法,子类的一切都运行正常。

1 个答案:

答案 0 :(得分:1)

Q& As说启用僵尸(或在Zombies乐器下运行),因为僵尸经常会导致这种错误。

但这不是唯一的原因。

您的程序正在破坏堆。最有可能的是它覆盖了自动释放池中某个对象中的isa指针(类指针),因此当运行循环消耗自动释放池时,objc_release会尝试取消引用伪造isa指针和崩溃。

启用僵尸可以屏蔽这样的错误,因为对于僵尸,运行时永远不会释放对象。这意味着你最终会得到很多不被使用的堆(除非你试图向僵尸发送消息),所以如果你破坏了它们,它们永远不会造成麻烦。

这些类型的崩溃可能很难调试,但有一个名为“guard malloc”的工具有时会有所帮助。从菜单栏中选择产品>方案>编辑方案。单击左侧列表中的“运行”操作。然后单击“诊断”选项卡。打开“启用Guard Malloc”选项。然后尝试重现崩溃。 Guard malloc会立即检测某些类型的堆损坏,并在破坏性指令上停止程序。