'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方法也是一样的。