13

自从在 iOS 8 上测试我的应用程序以来,我发现一个解决视图控制器初始化和演示的工作非常慢。

我曾经在 iOS 6 和 7 上使用过类似的代码:

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
    ....

    [self.window setRootViewController:_rootController];
    [self.window makeKeyAndVisible];

    // Conditions

    if (#first launch condition#) {
        // quite small controller containing Welcome showcase
        WelcomeViewController *w = ....
        [_rootViewController presentViewController:w animated:NO];
    }

    else if (#last opened item condition#) {
        // pretty big container, root view controller contains
        // a grid view which opens Item detail container the same way
        ItemDetailController *item = ....
        [_rootViewController presentViewController:item animated:NO];
    }

}

这在 iOS 8 中变成了一个非常缓慢的地狱。根视图控制器现在显示为 0.5-1 秒,然后立即用呈现的视图重绘屏幕。此外,演示的缓慢开始引起Unbalanced calls to begin/end appearance transitions _rootViewController警告。

最初的快速提示是通过调用另一个函数来移动这两个条件,并以零延迟调用它,以便在下一个主运行循环中处理它:

[self performSelector:@selector(postAppFinishedPresentation) withObject:nil afterDelay:0];

或类似的东西。这解决了不平衡的调用问题,但视觉差距(rootviewcontroller、gap、presented one)变得(显然)更大。

当您将通常称为以下内容时,演示文稿的缓慢也很明显:

// Example: Delegate caught finished Sign In dialog,
//          dismiss it and instantly switch to Profile controller

-(void)signInViewControllerDidFinishedSuccessfully
{
    [self dismissViewControllerAnimated:NO completion:^{
         UserProfileViewController *userProfile = ...
         [self presentViewController:userProfile animated:NO];
    }];
}

这应该是一段完全公平的代码,它曾经在 iOS 7 上执行直接转换而无需父视图控制器的可见轻弹。现在,同样的事情 - 过渡期间的父轻弹,即使它都在没有动画的情况下处理。

有人面临这个问题吗?有什么解决办法吗?我很想解决这个问题,而无需为UIWindow我需要完美传输的每一件事都用 s 做一些有趣的魔法。

4

4 回答 4

1

我不确定要求是否仅限于拥有根视图控制器并在那里显示任何内容。

但是根据您的代码,它有受欢迎的视图控制器,我认为在这种情况下,这种逻辑更有用。

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
    // Conditions

    if (#first launch condition#) {
        // quite small controller containing Welcome showcase
        WelcomeViewController *w = ....

        //It can be navigation or tab bar controller which have "w" as rootviewcontroller
        [self.window setRootViewController:w];
    }

    else if (#last opened item condition#) {
        // pretty big container, root view controller contains
        // a grid view which opens Item detail container the same way
        ItemDetailController *item = ....
        //It can be navigation or tab bar controller which have "item" as rootviewcontroller
        [self.window setRootViewController:item];
    }

    [self.window makeKeyAndVisible];

}
于 2015-03-09T13:18:58.683 回答
0

如果您使用 Storyboard,为什么不尝试:

UIStoryboard *storyboard = [UIStoryboard storyboardWithName:[[NSBundle mainBundle].infoDictionary objectForKey:@"UIMainStoryboardFile"] bundle:[NSBundle mainBundle]];
ViewController *_rootController = [storyboard instantiateViewControllerWithIdentifier:@"root"];
[self.window setRootViewController:_rootController];
[self.window makeKeyAndVisible];
if (vcToShow == 1) {
     ViewController2 *w = [storyboard instantiateViewControllerWithIdentifier:@"vc2"];
    [_rootController presentViewController:w animated:NO completion:nil];
}
else if (vcToShow == 2) {
    ViewController2 *w = [storyboard instantiateViewControllerWithIdentifier:@"vc3"];
    [_rootController presentViewController:w animated:NO completion:nil];
}

看起来这里没有延迟。

于 2014-10-25T15:11:51.540 回答
0

我从解雇/礼物对中得到的延迟是由此解决的。它可能会或可能不会帮助您的实例。我不得不完全改变我在 iOS 8 下显示/关闭模式视图控制器的策略:

[_rootViewController presentViewController:vc1 animated:NO completion:nil];

if(iNeedToDisplayVC2) {
     [vc1 presentViewController:vc2 animated:NO completion:nil];
}

因此,一旦我稍后完成了 vc2,我就会在同一个调用中同时关闭 vc1 和 vc2。这个策略也适用于 iOS 7。我也假设早期版本,但我没有测试它们。

于 2014-11-04T18:00:52.280 回答
0

在 iOS8 中,我发现旧的演示文稿在完成块中还没有完全完成,立即调用dismiss 或 present 会导致控制台消息,有时甚至不会出现 present/dismiss。通过在第二个演示文稿中添加进一步延迟的表演,我取得了一些成功:

 [self dismissViewControllerAnimated:NO completion:^{
         UserProfileViewController *userProfile = ...
         [[NSOperationQueue mainQueue] addOperationWithBlock:^{
             [self presentViewController:userProfile animated:NO];
         }];
    }];
于 2015-01-27T22:04:52.040 回答