我发现了一个僵尸..现在怎么办?

I've detected a zombie.. now what?

我的应用程序中有以下导航流程:

视图控制器 A 嵌入在导航控制器中。
A 模态地呈现 B。 在用户关闭 B 之前,导航控制器的视图控制器设置为:[A, C] B 被解雇后,C 出现在屏幕上。

以下是从 B 设置视图控制器的方式:

let appDelegate  = UIApplication.sharedApplication().delegate as! AppDelegate
let rootViewController = appDelegate.window!.rootViewController as! RootNavigationController

let Ccontroller = UIViewController()

var newControllers = rootViewController.viewControllers

newControllers.append(Ccontroller)
rootViewController.viewControllers = newControllers

dispatch_async(dispatch_get_main_queue(), { () -> Void in
  self.dismissViewControllerAnimated(true, completion: nil)
})

按下控制器C上的后退按钮后检测到僵尸。崩溃不一致。有时我浏览一次导航流程,有时我浏览这些步骤 15 次后才收到崩溃。

我查看了所有与在已释放对象和僵尸上调用函数相关的问题,但大部分与 Objective-C 中的 retain/release 有关。我正在寻找 Swift 解决方案。

最常见的崩溃日志:

*** -[CALayer retain]: message sent to deallocated instance 0x14b7d8880

我们的测试人员发现了其他几个崩溃日志,我认为它们与同一个僵尸问题有关。

    Thread : Crashed: com.apple.main-thread
0  libobjc.A.dylib                0x180a780b0 objc_retain + 16
1  UIKit                          0x186617c88 -[UIImageView isAnimating] + 132
2  UIKit                          0x186605630 -[UIImageView stopAnimating] + 112
3  UIKit                          0x186605334 -[UIImageView dealloc] + 64
4  libobjc.A.dylib                0x180a5eb54 object_cxxDestructFromClass(objc_object*, objc_class*) + 148
5  libobjc.A.dylib                0x180a6a040 objc_destructInstance + 92
6  libobjc.A.dylib                0x180a6a0a0 object_dispose + 28
7  UIKit                          0x1869bae80 -[UIResponder dealloc] + 140
8  UIKit                          0x186604ce8 -[UIView dealloc] + 1436
9  UIKit                          0x18677fee0 -[UIControl dealloc] + 72
10 UIKit                          0x18677ffe8 -[UIButton dealloc] + 100
11 UIKit                          0x186a59474 -[UITouch .cxx_destruct] + 136
12 libobjc.A.dylib                0x180a5eb54 object_cxxDestructFromClass(objc_object*, objc_class*) + 148
13 libobjc.A.dylib                0x180a6a040 objc_destructInstance + 92
14 libobjc.A.dylib                0x180a6a0a0 object_dispose + 28
15 UIKit                          0x186643388 -[UITouch dealloc] + 72
16 CoreFoundation                 0x18142e384 __CFBasicHashDrain + 276
17 CoreFoundation                 0x1812e2bdc CFDictionaryRemoveAllValues + 352
18 UIKit                          0x186a51310 -[UITouchesEvent _setHIDEvent:] + 136
19 UIKit                          0x186606dd8 _UIApplicationHandleEventQueue + 3984
20 CoreFoundation                 0x1813b0538 __CFRUNLOOP_IS_CALLING_OUT_TO_A_SOURCE0_PERFORM_FUNCTION__ + 24
21 CoreFoundation                 0x1813affcc __CFRunLoopDoSources0 + 540
22 CoreFoundation                 0x1813adccc __CFRunLoopRun + 724
23 CoreFoundation                 0x1812d9670 CFRunLoopRunSpecific + 384
24 GraphicsServices               0x182bb0088 GSEventRunModal + 180
25 UIKit                          0x186671ec4 UIApplicationMain + 204
26 Slide                          0x100062a0c main (AppDelegate.swift:14)
27 libdispatch.dylib              0x180e768b8 (Missing)

_

Thread : Crashed: com.apple.main-thread
0  QuartzCore                     0x24bfe690 CA::Layer::model_layer(CA::Transaction*) + 51
1  QuartzCore                     0x24bfe64d -[CALayer modelLayer] + 32
2  QuartzCore                     0x24bfe64d -[CALayer modelLayer] + 32
3  QuartzCore                     0x24bfe4e9 -[CALayer animationForKey:] + 56
4  UIKit                          0x26b0fbb5 -[UIImageView isAnimating] + 136
5  UIKit                          0x26afdee3 -[UIImageView stopAnimating] + 102
6  UIKit                          0x26afdc33 -[UIImageView dealloc] + 66
7  libobjc.A.dylib                0x22169f55 object_cxxDestructFromClass(objc_object*, objc_class*) + 116
8  libobjc.A.dylib                0x22173e47 objc_destructInstance + 34
9  libobjc.A.dylib                0x22173e6b object_dispose + 14
10 UIKit                          0x26e99ccd -[UIResponder dealloc] + 128
11 UIKit                          0x26afd635 -[UIView dealloc] + 1428
12 UIKit                          0x26c7368d -[UIControl dealloc] + 64
13 UIKit                          0x26c73789 -[UIButton dealloc] + 96
14 UIKit                          0x26f2c2ab -[UITouch .cxx_destruct] + 126
15 libobjc.A.dylib                0x22169f55 object_cxxDestructFromClass(objc_object*, objc_class*) + 116
16 libobjc.A.dylib                0x22173e47 objc_destructInstance + 34
17 libobjc.A.dylib                0x22173e6b object_dispose + 14
18 UIKit                          0x26b3ae99 -[UITouch dealloc] + 64
19 libobjc.A.dylib                0x22184f67 objc_object::sidetable_release(bool) + 150
20 CoreFoundation                 0x229f0058 __CFBasicHashDrain + 336
21 CoreFoundation                 0x228e1aa5 CFDictionaryRemoveAllValues + 276
22 UIKit                          0x26f264a5 -[UITouchesEvent _setHIDEvent:] + 120
23 UIKit                          0x26aff2ef _UIApplicationHandleEventQueue + 3366
24 CoreFoundation                 0x2298768f __CFRUNLOOP_IS_CALLING_OUT_TO_A_SOURCE0_PERFORM_FUNCTION__ + 14
25 CoreFoundation                 0x2298727d __CFRunLoopDoSources0 + 452
26 CoreFoundation                 0x229855eb __CFRunLoopRun + 794
27 CoreFoundation                 0x228d8bf9 CFRunLoopRunSpecific + 520
28 CoreFoundation                 0x228d89e5 CFRunLoopRunInMode + 108
29 GraphicsServices               0x23b24ac9 GSEventRunModal + 160
30 UIKit                          0x26b68ba1 UIApplicationMain + 144
31 Slide                          0xccfbc main (AppDelegate.swift:14)
32 libdispatch.dylib              0x22587873 (Missing)

您将如何确定导致此问题的代码?

通常当您的 ViewController 被释放时会发生此崩溃。避免此崩溃的最佳做法是将 ViewController 设为 属性。为它编写getter方法。

@property (nonatomic, strong) ViewController *aViewController;

(ViewController*)aViewController{

     if(_aViewController == nil){
           UIStoryboard *mainStoryboard = [UIStoryboard storyboardWithName:@"MainStoryboard" bundle: nil];

           _aViewController = (aViewController*)[mainStoryboard 
                    instantiateViewControllerWithIdentifier: @"ViewControllerIdentifier"];
      }
      return aViewController;
}


 - (IBAction)pushViewController{
        [self.navigationController pushViewController:self.aViewController animated:YES];
}

避免在导航方法中创建 ViewController 的实例。只需做一个参考并一次又一次地使用它。在您的导航方法中一次又一次地进行新引用时,您以前的 viewcontroller 实例可能会被释放。