iOS连续dismiss几个ViewController的方法,以及切换根视图我遇到的问题

背景:登录注册界面有控制器A、B、C,登录后切换根视图为主页D

iOS连续dismiss几个ViewController的方法,以及切换根视图我遇到的问题_第1张图片
屏幕快照 2016-07-16 下午4.22.08.png

切换根视图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 ?视图销毁后它的视图不应该就没有了吗,实在是想不通。

效果图如下
iOS连续dismiss几个ViewController的方法,以及切换根视图我遇到的问题_第2张图片
Untitled.gif

开始我以为是代码写的有问题视图没有被释放掉,后来新建一个项目专门这样切换根视图,发现有同样的问题,反复试验后发现:

  • 如果我在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,发现这样可以达到效果;

iOS连续dismiss几个ViewController的方法,以及切换根视图我遇到的问题_第3张图片
Untitled2.gif

我不知道为什么会出现A控制器销毁后还会在根视图D上保持显示这种现象,如果有人知道欢迎告诉我~

另外,如果dismiss动画设置为YES,会神奇的发现根视图切换为D时,能看到C(蓝色)、B(黄色)、A(红色)依次消失:

iOS连续dismiss几个ViewController的方法,以及切换根视图我遇到的问题_第4张图片
Untitled3.gif

补充:如果是UINavigationController

今天看MJRefresh的demo,其中使用的是UINavigationController来push控制器,发现demo中切换根视图时没有进行pop操作,导航控制器栈内的所有控制器都dealloc被释放了。然后我试验了下的确如此,虽然我不明白为啥模态和导航控制器,同样是根视图的引用被切换释放了,却产生不同效果。我猜测可能是导航控制器和模态跳转的一个不同吧... 以后有多层跳转后切换根视图就尽量用导航控制器去做了。

你可能感兴趣的:(iOS连续dismiss几个ViewController的方法,以及切换根视图我遇到的问题)