問題描述:這個問題非常常見,就是平時我們做一個按鈕(我們假設這個頁面是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方法也是一樣的。