美文网首页Swifty CodingiOS开发小技巧fish的iOS
iOS连续dismiss几个ViewController的方法,

iOS连续dismiss几个ViewController的方法,

作者: plantseeds | 来源:发表于2016-07-15 14:31 被阅读4050次

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

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

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

    Untitled2.gif

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

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

    Untitled3.gif

    补充:如果是UINavigationController

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

    相关文章

      网友评论

      • DarrenG:作者在切换跟控制器的时候没有遇到会有一瞬间的黑屏这种问题么?
      • 7f5708eec131:前两天看到一个方法,设置一个全局BOOL变量“closeAll”,在每一层viewWillAppear中判断,当closeAll==YES时dismiss,因此在最后一层控制器dismiss之前把closeAll属性赋值为YES就可以实现连续跳转了。

      本文标题:iOS连续dismiss几个ViewController的方法,

      本文链接:https://www.haomeiwen.com/subject/jzyyjttx.html