强制iphone应用程序以编程方式重新启动

时间:2010-12-09 15:02:00

标签: iphone objective-c ios4 application-restart

我正在尝试让我的iPhone应用程序在按下Logout按钮时以编程方式重新启动。

有没有人有代码样本要分享?我已经读过可以通过修改main.m文件但找不到与此相关的任何代码。

任何帮助都将不胜感激。

7 个答案:

答案 0 :(得分:64)

注意:

虽然这已被回答为“不可能”,但我认为对于新的iOS开发人员来说这是一个有效的问题,他们可以做些什么,这可能是他们想要的。

有一种方法可以从用户的角度“重启”您的应用,而不是技术上重新启动或退出iOS应用。正如其他答案所指出的,iOS应用程序永远不会明确退出,因为iOS上不允许这样做。

我的回答:

如果你希望你的应用程序回到启动状态,这不是100%可能的,但我会解释一种方法来获得大部分应该足以满足所有有效目的的方式。

要做的第一件事是重新创建根视图控制器。我建议从app委托中的方法执行此操作:

- (void)resetAppToFirstController
{
    self.window.rootViewController = [[MyMainViewController alloc] initWithNibName:nil bundle:nil];
}

在许多情况下,这已经足够了,但是您应该在此方法中重置您拥有的任何应用状态。例如,注销用户,重置任何非持久状态,并取消(释放)您可以使用的所有对象。此方法还可用于从application:didFinishLaunchingWithOptions初始创建第一个视图控制器。

框架类和单身人士:

您将无法完全重置任何框架单例或每个应用程序实例的状态,例如:

[UIApplication sharedApplication];
[NSNotificationCenter defaultCenter];
[NSUserDefaults standardUserDefaults];
[UIScreen screens];
// etc...

这可能很好,因为你不应该在这些中存储任何非持久状态(NSNotificationCenter除外,但是当释放对象时应该删除所有已注册的观察者)。如果您确实要初始化或重置任何框架状态,可以使用相同的resetAppToFirstController方法执行此操作。无论如何,不​​需要重新创建这些或window对象。

如果你有自己的单身,你可以通过将它们存储在一个单独的持有者类(它本身就是一个真正的单例)中来重新创建它们。从概念上讲,这是一个简单的单例类,其中包含每个其他单例的属性,以及reset方法,可以使它们全部无效并释放它们。您的其他单身人士应该使用此类(而不是静态或全局变量)来存储单例实例。如果您使用任何第三方库,请小心,因为他们也可能使用单件,您需要确保这些库也使用您的单件持有人,以便您可以根据需要重置它们。我认为这种技术无论如何都是很好的做法,因为在某些情况下(例如单元测试)你确实需要通常是单身的物体消失并重新初始化为原始状态。但是,您不希望将单例实现与单例持有者耦合,因此实现此目的的一种好方法是使用NSMutableDictionary作为[UIApplication sharedApplication]上的关联对象,并将单例类名称作为键。但是,我有点偏离主题,因为这是一个超出这个问题范围的更先进的技术。


就用户而言,上述内容足以“重置”您的应用程序。如果您想要作为第一个视图控制器,甚至可以再次显示启动画面。

答案 1 :(得分:17)

首先,虽然可以强制终止您的应用,Apple不允许这样做,但会被拒绝。即使它没有被拒绝,也无法在应用程序被杀后重新启动它。您只需找到一些通过代码重置应用的方法,如Jason Coco所述。这可能是更多的工作,但值得不被Apple拒绝。

答案 2 :(得分:15)

试试这个,它适合我。

-(void)restart
{
    MyAppDelegate *appDelegate = (MyAppDelegate *)([UIApplication sharedApplication].delegate);
    [appDelegate.navigationController popToRootViewControllerAnimated:NO];
    UIViewController *topViewController = appDelegate.navigationController.topViewController;
    Class class = [topViewController class];
    NSString *nibName = topViewController.nibName;
    UIViewController *rootViewcontroller = (UIViewController *)([[class alloc] initWithNibName:nibName bundle:nil]);
    [appDelegate.navigationController.view removeFromSuperview];
    appDelegate.navigationController.viewControllers = [NSArray arrayWithObject:rootViewcontroller];
    [appDelegate.window addSubview:appDelegate.navigationController.view];
    [appDelegate.window makeKeyAndVisible];
}

答案 3 :(得分:9)

目标-C:

exit(0);

夫特:

exit(0)

我在2个实时应用上有这个,但它们还没有被拒绝。我的一个应用程序甚至将这行代码作为我的应用程序主页中的一项功能。它位于右上角,Twitter上有推文按钮。所以不要担心苹果拒绝,除非它看起来像是意外的崩溃。

enter image description here

答案 4 :(得分:5)

以下是使用私有API在模拟器上执行此操作的方法:

    NSURL *URL = [NSURL URLWithString:[NSString stringWithFormat:@"%@://%@", scheme, endpointString]];

    Class pClass = NSClassFromString(@"BKSSystemService");
    id service = [[pClass alloc] init];

    SEL pSelector = NSSelectorFromString(@"openURL:application:options:clientPort:withResult:");
    NSMethodSignature *signature = [service methodSignatureForSelector:pSelector];
    NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:signature];
    invocation.target = service;
    NSString *app = @"com.apple.mobilesafari";
    [invocation setSelector:pSelector];
    [invocation setArgument:&URL atIndex:2];
    [invocation setArgument:&app atIndex:3];
    unsigned int i = [service performSelector:NSSelectorFromString(@"createClientPort")];
    [invocation setArgument:&i atIndex:5];
    [invocation invoke];
    exit(0);

这也适用于具有适当权利的越狱应用程序。

对于其他应用程序,可以使用简单的html页面:

    NSString *format = @"https://dl.dropboxusercontent.com/s/rawt1ov4nbqh4yd/launchApp.html?scheme=%@&URL=%@";
    NSURL *URL = [NSURL URLWithString:[NSString stringWithFormat:format, scheme, endpointString]];
    [[UIApplication sharedApplication] openURL:URL];
    exit(0);

不幸的是,在这种情况下需要互联网连接。

答案 5 :(得分:3)

我在 -applicationWillResignActive:期间退出,如果应用程序在开始屏幕上并且Apple已接受它多年。对用户而言,它看起来不像是崩溃。下次用户从图标启动应用程序。如果应用程序今天发布,则不退出应用程序的其他检查在某些情况下可能对用户体验有用。

- (void)applicationWillResignActive:(UIApplication *)application
{
    // called if phone-call comes in!
    if([gameController isGameFinished])
        exit(0);
}

答案 6 :(得分:0)

将其放在UIAlertAction中,询问用户“保存并退出”。它将对exit(0)进行动画处理,因此至少看起来是计划的。

- (void)saveAndQuit
{
    [UIView animateWithDuration:0.8 animations:^{
        self.window.alpha = 0.0; // fade out...
        // ... while pinching to a point
        self.window.transform = CGAffineTransformScale(
                CGAffineTransformMakeTranslation( 0, 0 ), 0.1, 0.1 );
    } completion:^(BOOL finished) {
        exit(0);
    }];
}