关于项目中崩溃问题处理:Can't add self as subview

项目中bugly总是收集到Can't add self as subview 的崩溃
错误,崩溃调用堆栈解析如下:

图:
3D520C11-0552-41A6-8FB1-442E0AEC48C1.png

通过分析崩溃堆栈日志解析和崩溃信息提示,只能得出两个线索:
1.addSubView的参数是自己本身self;
2.崩溃和navigationController转场动画有关。

排查

1.addSubView的参数是自己本身self

在工程中用 [self addSubView:self] 测试,确实会崩溃,崩溃的堆栈信息如下。和上传的崩溃信息不一致,可以排除。


屏幕快照 2019-02-12 18.05.08.png

2.navigationController转场动画

navigationController push或者pop上一次操作还没有完成就开始执行下一次操作,同步执行了多个转场操作。如下代码:

    [self.navigationController pushViewController:[[SecondAViewController alloc] init] animated:NO];
    [self.navigationController pushViewController:[[SecondBViewController alloc] init] animated:YES];
    [self.navigationController pushViewController:[[SecondCViewController alloc] init] animated:YES];

同时执行完以上操作,之后的pop退场操作就会导致崩溃。崩溃信息如下:


屏幕快照 2019-02-13 16.37.35.png

push SecondCViewController,C成功入栈,但是视图没有加载到容器中,实际显示的还是B的vc与view,但是栈顶是C的vc。

第一次点击返回时,实际是C出栈,但是当前显示的视图B的view被先加载到formAnimateView与toAnimateView上,原本视图在出栈后应该被释放,但是现在容器栈内还存在B的vc;
当第二次点击返回时,实际应该是B的vc出栈,但是A的view加载到toAnimateView上之后,toAnimateView需要加载到wrapperView进行transition动画, 但wrapperView通过栈顶元素view.superview取值, 而栈顶元素B的view由于上一次错误的转场, 并未在transition动画完成后挂载到wrapperView, 还保留在的临时的动画视图toAnimateView上, 所以使toAnimateView加载到WrapperView的操作变成了动画视图toAnimateView加载到自己上

解决方法

导致转场异常的根本原因是上一次操作还没执行结束就开始执行下一次操作, 同步执行了多个转场操作,这时就需要拦截控制器入栈\出栈的方法,确保当有控制器正在进行入栈\出栈的操作时,没有其他入栈\出栈操作。

实现

通过Runtime的方法魔法Method Swizzling技术实现。分类实现修改navigationControlle的pop和push方法。在push\pop方法中设置一个标志位,动画结束之后,重置标志位,通过标志位来判断push\pop操作是否执行。代码实现如下:

@interface UINavigationController () <UINavigationControllerDelegate>

@property (nonatomic, assign) BOOL viewTransitionInProgress;

@end

@implementation UINavigationController (SafeTransition)

+ (void)load {
    method_exchangeImplementations(class_getInstanceMethod(self, @selector(pushViewController:animated:)),
                                   class_getInstanceMethod(self, @selector(safePushViewController:animated:)));
    
    method_exchangeImplementations(class_getInstanceMethod(self, @selector(popViewControllerAnimated:)),
                                   class_getInstanceMethod(self, @selector(safePopViewControllerAnimated:)));
    
    method_exchangeImplementations(class_getInstanceMethod(self, @selector(popToRootViewControllerAnimated:)),
                                   class_getInstanceMethod(self, @selector(safePopToRootViewControllerAnimated:)));
    
    method_exchangeImplementations(class_getInstanceMethod(self, @selector(popToViewController:animated:)),
                                   class_getInstanceMethod(self, @selector(safePopToViewController:animated:)));
    
}

#pragma mark - setter & getter
- (void)setViewTransitionInProgress:(BOOL)property {
    NSNumber *number = [NSNumber numberWithBool:property];
    objc_setAssociatedObject(self, @selector(viewTransitionInProgress), number, OBJC_ASSOCIATION_RETAIN);
}

- (BOOL)viewTransitionInProgress {
    NSNumber *number = objc_getAssociatedObject(self, @selector(viewTransitionInProgress));
    return [number boolValue];
}

#pragma mark - Intercept Pop, Push, PopToRootVC
- (NSArray *)safePopToRootViewControllerAnimated:(BOOL)animated {
    if (self.viewTransitionInProgress) return nil;
    
    if (animated) {
        self.viewTransitionInProgress = YES;
    }
    
    NSArray *viewControllers = [self safePopToRootViewControllerAnimated:animated];
    if (viewControllers.count == 0) {
        self.viewTransitionInProgress = NO;
    }
    
    return viewControllers;
}

- (NSArray *)safePopToViewController:(UIViewController *)viewController animated:(BOOL)animated {
    if (self.viewTransitionInProgress) return nil;
    
    if (animated){
        self.viewTransitionInProgress = YES;
    }
    
    NSArray *viewControllers = [self safePopToViewController:viewController animated:animated];
    if (viewControllers.count == 0) {
        self.viewTransitionInProgress = NO;
    }
    
    return viewControllers;
}

- (UIViewController *)safePopViewControllerAnimated:(BOOL)animated {
    if (self.viewTransitionInProgress) return nil;
    
    if (animated) {
        self.viewTransitionInProgress = YES;
    }
    
    UIViewController *viewController = [self safePopViewControllerAnimated:animated];
    if (viewController == nil) {
        self.viewTransitionInProgress = NO;
    }
    
    return viewController;
}

- (void)safePushViewController:(UIViewController *)viewController animated:(BOOL)animated {
    if (self.viewTransitionInProgress == NO) {
        [self safePushViewController:viewController animated:animated];
        
        if (animated) {
            self.viewTransitionInProgress = YES;
        }
    }
}

@end

@implementation UIViewController (SafeTransitionLock)

+ (void)load {
    Method m1;
    Method m2;
    
    m1 = class_getInstanceMethod(self, @selector(safeViewDidAppear:));
    m2 = class_getInstanceMethod(self, @selector(viewDidAppear:));
    
    method_exchangeImplementations(m1, m2);
}

- (void)safeViewDidAppear:(BOOL)animated {
    self.navigationController.viewTransitionInProgress = NO;
    
    [self safeViewDidAppear:animated];
}

@end

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 转场过程解析 UINavigationController对于translation动画做了一定的封装, 同时持有...
    Archerlly阅读 6,736评论 24 10
  • 前言的前言 唐巧前辈在微信公众号「iOSDevTips」以及其博客上推送了我的文章后,我的 Github 各项指标...
    VincentHK阅读 5,414评论 3 44
  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,135评论 1 32
  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 12,192评论 4 61
  • 其实,阅读《商君书》这样的经典文章,除了要了解当时的社会背景、发展情况,更要结合当时的社会实际来分析和把握,这样才...
    健康体验阅读 421评论 0 3