背景:登录注册界面有控制器A、B、C,登录后切换根视图为主页D
切换根视图C->D的时候,我直接用的下面这句代码
[UIApplication sharedApplication].delegate.window.rootViewController = [ViewControllerD new];
发现没有进入A、B、C相应的dealloc方法,就是说控制器A、B、C并没有被释放。检查后发现是因为我切换根视图控制器前,忘了dismiss控制器A、B、C。
该怎么dismiss掉3个控制器呢?
百度查到:通过self.presentingViewController可以获取到当前控制器的父控制器,self.presentingViewController.presentingViewController获取父父控制器 and so on...,所以我们需要通过代理或者通知来一层层的dismiss吗?很显然这并不是一个好方法。
这篇文章提供了新的思路 ios 如何dismiss连续好几个viewControllers:
查了苹果官网发现:
If you present several view controllers in succession, thus building a stack of presented view controllers, calling this method on a view controller lower in the stack dismisses its immediate child view controller and all view controllers above that child on the stack. When this happens, only the top-most view is dismissed in an animated fashion; any intermediate view controllers are simply removed from the stack. The top-most view is dismissed using its modal transition style, which may differ from the styles used by other view controllers lower in the stack.
大致就是如果您连续弹出几个视图控制器,从而为这些弹出的视图控制器建立一个堆栈,调用dismissViewControllerAnimated:completion:这个方法去解散其他的视图控制器。仅仅是最顶层是被解散,中间的视图控制器从栈中删除了。
并且文中提供了用代理或者通知去dismiss最顶层控制器的方法,但是我觉得有些麻烦,既然可以获取到父控制器,应该可以用循环实现,所以我用while循环实现如下:
//把最底部的视图控制器dismiss掉
UIViewController *parentVC = self.presentingViewController;
UIViewController *bottomVC;
while (parentVC) {
bottomVC = parentVC;
parentVC = parentVC.presentingViewController;
}
[bottomVC dismissViewControllerAnimated:NO completion:nil];
[UIApplication sharedApplication].delegate.window.rootViewController = [ViewControllerD new];
这样就可以在当前视图中dismiss最底部控制器了,并且进入dealloc方法A、B、C控制器都销毁了.(如果我这样写有问题欢迎指出)
但是发现了新的问题
由于D中没有任何东西,甚至连背景色都没添加!所以在切换根视图控制器后,发现D视图上居然显示出最底部A视图的界面!再确认根控制器确实是D并且A、B、C都进入dealloc中销毁掉后,WTF ?视图销毁后它的视图不应该就没有了吗,实在是想不通。
效果图如下
开始我以为是代码写的有问题视图没有被释放掉,后来新建一个项目专门这样切换根视图,发现有同样的问题,反复试验后发现:
- 如果我在D上添加了背景色或者view,它就会把A视图的view给挡住;
- 如果只有一个控制器A,直接从A切换根视图到D,那么不会出现这种问题,甚至切换根视图前都不用dismiss掉A,但如果有A->B多个的话就会出现我这种问题;
- D上显示的 A中view的约束会出错;
- D上显示的 A中的button是不可点击的;
这样写可以达到效果
百思不得其解后,最终我把目光投到了 [dismissViewControllerAnimated: completion:] 方法的completion block中,我是想在视图dismiss完成后再切换根视图,所以我把切换根视图的操作放到了 dismiss的completion块中:
//把最前面的视图控制器dismiss掉
UIViewController *parentVC = self.presentingViewController;
UIViewController *bottomVC;
while (parentVC) {
bottomVC = parentVC;
parentVC = parentVC.presentingViewController;
}
[bottomVC dismissViewControllerAnimated:NO completion:^{
//dismiss后再切换根视图
[UIApplication sharedApplication].delegate.window.rootViewController = [TabBarController new];
}];
return;
将dismiss动画Animated设置为NO,发现这样可以达到效果;
我不知道为什么会出现A控制器销毁后还会在根视图D上保持显示这种现象,如果有人知道欢迎告诉我~
另外,如果dismiss动画设置为YES,会神奇的发现根视图切换为D时,能看到C(蓝色)、B(黄色)、A(红色)依次消失:
补充:如果是UINavigationController
今天看MJRefresh的demo,其中使用的是UINavigationController来push控制器,发现demo中切换根视图时没有进行pop操作,导航控制器栈内的所有控制器都dealloc被释放了。然后我试验了下的确如此,虽然我不明白为啥模态和导航控制器,同样是根视图的引用被切换释放了,却产生不同效果。我猜测可能是导航控制器和模态跳转的一个不同吧... 以后有多层跳转后切换根视图就尽量用导航控制器去做了。