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

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容