今天不经意间发现一个好玩的问题,感觉这个问题挺奇葩的,而且以前没有遇到过,然后百度发现就有两个回答,还是老外的回答
http://stackoverflow.com/questions/15287678/warning-attempt-to-present-viewcontroller-whose-view-is-not-in-the-window-hiera
http://stackoverflow.com/questions/26022756/warning-attempt-to-present-on-whose-view-is-not-in-the-window-hierarchy-s#
问题的起因在于,我在viewDidLoad方法中直接加载另一个viewcontroller。起初我把tap方法写在了viewDidLoad方法中,然后运行,发现控制台打印了
2016-07-07 15:26:44.740 CALayerFun[2492:255155] Warning: Attempt to present <UIAlertController: 0x7f995360dd40> on <LCSKeyBoardViewController: 0x7f99537aacb0> whose view is not in the window hierarchy!
这样一段话,当然模拟器也没有加载这个alert成功,没有反应。
这是什么情况,首先检查代码,代码和平时写的一样,没问题啊。唯一的不同就是以前都是通过一个点击也好,其他方式也好,来唤起这个alert。但是今天我直接在viewDidLoad中加载的这个alert。可能问题出在这。
首先看看原来的UIAlertView是不是和现在使用的UIAlertController结果一样,然后直接写了一个UIAlertView在viewDidLoad方法中。这时候运行程序,发现正常加载。。。这说明什么,这说明问题出在一个是view和一个是viewController。然后,本着眼见为实的目的,新建一个viewController也让它在viewDidLoad方法中跳转,这个时候发现和跳转UIAlertController有一样的错误,跳转不成功,打印信息也一样。
先附上代码,都是日常代码,没有任何技术含量 ~~
- (void)viewDidLoad {
[superviewDidLoad];
self.view.backgroundColor = [UIColorwhiteColor];
//第一种,用原来的alertview(正常)
UIAlertView *alert = [[UIAlertViewalloc]initWithTitle:@"提示"message:@" 新闻要点,最新资讯,及时更新,新闻要点,最新资讯,及时更新新闻要点,最新资讯,及时更新新闻要点,最新资讯,及时更新新闻要点"delegate:nilcancelButtonTitle:@"好的"otherButtonTitles:@"不要",nil];
[alert show];
return;
//第二种,直接上来就调用(不正常)
[selftap];
return;
//第三种,经过四秒后调用(正常)
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(4 *NSEC_PER_SEC)),dispatch_get_main_queue(), ^{
[selftap];
});
return;
//第四种,通过点击事件调用(正常)
UITapGestureRecognizer *tap = [[UITapGestureRecognizeralloc]initWithTarget:selfaction:@selector(tap)];
[self.viewaddGestureRecognizer:tap];
return;
}
- (void)tap{
UIAlertController *alert = [UIAlertControlleralertControllerWithTitle:@"提示"message:@" 新闻要点,最新资讯,及时更新,新闻要点,最新资讯,及时更新新闻要点,最新资讯,及时更新新闻要点,最新资讯,及时更新新闻要点"preferredStyle:UIAlertControllerStyleAlert];
UIAlertAction *yesAction = [UIAlertActionactionWithTitle:@"好的"style:UIAlertActionStyleDefaulthandler:^(UIAlertAction *_Nonnull action) {
NSLog(@"yes");
}];
UIAlertAction *noAction = [UIAlertActionactionWithTitle:@"不要"style:UIAlertActionStyleCancelhandler:^(UIAlertAction *_Nonnull action) {
NSLog(@"no");
}];
[alert addAction:yesAction];
[alert addAction:noAction];
[selfpresentViewController:alertanimated:YEScompletion:^{
}];
}
2016-07-07 15:40:35.734 CALayerFun[2636:268808] Warning: Attempt to present <ViewController: 0x7fb3017147a0> on <LCSKeyBoardViewController: 0x7fb30145c890> whose view is not in the window hierarchy!
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0*NSEC_PER_SEC)),dispatch_get_main_queue(), ^{
[selftap];
});
1.performSelector方法
此方式要求必须在主线程中执行,否则无效。是一种非阻塞的执行方式,暂时未找到取消执行的方法。
2.定时器:NSTimer
此方式要求必须在主线程中执行,否则无效。是一种非阻塞的执行方式,可以通过NSTimer类的- (void)invalidate;取消执行。
3. sleep方式
此方式在主线程和子线程中均可执行。是一种阻塞的执行方式,建方放到子线程中,以免卡住界面,没有找到取消执行的方法。
4.GCD方式
此方式在可以在参数中选择执行的线程。是一种非阻塞的执行方式,没有找到取消执行的方法。