• 'Can't add self as subview'崩溃日志详解


    问题描述:这个问题非常常见,就是平时我们做一个按钮(我们假设这个页面是RootVC),按钮加一个事件,点击这个事件的时候会push出一个新的控制器A,当我们连续快速(时间间隔在0.5S内,也就是PUSH前一个事件的PUSH动画还没结束之前)点击两次这个按钮的时候,就会导致这个按钮连续响应了两次事件,同时推出了两个控制器A1、A2(这两个控制器都是A类型的),当我们再次点击A1(A2)返回的时候,点击第一次返回会是黑屏,再次点击A2(A1)返回的时候,就会报以下这个崩溃。
    *** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: 'Can't add self as subview'

    问题重现:为了让这个问题重现,我们用以下方法让A1控制器push之后的0.3秒 我们再push出A1控制器:

     [self.navigationController pushViewController:controller animated:YES];//推出第一个控制器A1(查阅相关资料,push动画持续的时间大约为0.5S,为保证在push动画结束之前push第二个页面,我们使用0.3秒后push出第二个控制器)

                dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(.3 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
                    [self.navigationController pushViewController:[[MYController alloc] initWithNibName:@"MYController" bundle:nil] animated:YES];//0.3秒后push出A2控制器
                });
    进入到新的页面之后,我们按返回按钮,让视图控制器A1/A2pop出来,按照上面的问题描述,点击第一次返回的时候,我们看到队导航条之后都是黑屏,再按第二次返回的时候,我们会捕捉到'Can't add self as subview'这样的一个崩溃消息。

    问题解决方案:
    以前遇到类似的问题的时候,我们通常是让按钮接受到一个事件后,先让这个按钮的enable属性为NO,非使能,然后1秒之后再把按钮的enable属性为YES,这样通过控制连续快速点击来达到不会让同一个控制器RootVC同时推出同一个类型的控制器的两个对象A1/A2.很明显这是不明知的做法。

    现在我们给UINavigationController写一个类别(UINavigationController+GONavigationController.h"),分别实现以下四个方法:

    - (void)pushViewController:(UIViewController *)viewController animated:(BOOL)animated navigationLock:(id)lock;
    {
        if (!lock || self.topViewController == lock)
        {
            [self pushViewController:viewController animated:animated];
        }
    }
    - (id)navigationlock
    {
        return self.topViewController;
    }
    - (NSArray *)popToViewController:(UIViewController *)viewController animated:(BOOL)animated navigationLock:(id)lock;\
    {
        if (!lock || self.topViewController == lock)
        {
            [self popToViewController:viewController animated:animated];
        }
         return @[];
    }
    - (NSArray *)popToRootViewControllerAnimated:(BOOL)animated navigationLock:(id)lock
    {
        if (!lock || self.topViewController == lock)
        {
            [self popToRootViewControllerAnimated:animated];
        }
        return @[];
    }


    调用的时候, 我们这么写:
    id lock = [self.navigationController navigationlock];
     [self.navigationController pushViewController:controller animated:YES navigationLock:lock];
    分析一下代码:这个时候lock就是我们的RootVC,一但我们调用这个方法,lock是与self.topViewController相等的,所以if分支成立,RootVC会正常推出A1(也就是controller),一旦推出A1,self.topViewController就变成了刚刚推出的A1控制器,
    这个时候我们再按照问题重现的方法写出以下代码 :
     dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(.3 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
                    MYController *controler2 = [[MYController alloc] initWithNibName:@"MYController" bundle:nil];
                    [self.navigationController pushViewController:controler2 animated:YES navigationLock:lock];
                });
    这里再次调用push的时候,self.topViewController已经为A1,而入参lock仍然为RootVC,if分支不成立,就会自动不再推出同一个类型的另一个对象A2(也就是controler2)

    同样的POP方法也是一样的。

  • 相关阅读:
    4-18
    Vue学习 2017-4-9
    前端杂谈
    不错的博客哦!
    待整理知识杂项
    Vue学习历程
    王工的权限理解
    【NX二次开发】图标图像
    【转】C++怎么读写windows剪贴板的内容?比如说自动把一个字符串复制.
    获取计算机名
  • 原文地址:https://www.cnblogs.com/yanyan1119/p/3951976.html
Copyright © 2020-2023  润新知