我正在编写一个iPhone应用程序,我需要强制它退出,因为某些用户操作。清理内存后分配的应用程序,调用终止应用程序的适当方法是什么?
你试过exit(0)
吗?
或者,[[NSThread mainThread] exit]
,虽然我没有尝试过它似乎是更合适的解决方案。
我的应用程序最近被拒绝了,我使用了一种无证的方法。从字面上看:
“不幸的是,它无法添加到App Store,因为它使用的是私有API。禁止使用非公开API,如iPhone开发人员计划许可协议第3.3.1节所述:
“3.3.1应用程序只能以Apple规定的方式使用Documented API,不得使用或调用任何私有API。”
应用程序中包含的非公共API是terminateWithSuccess“
Apple说:
“警告:不要调用退出函数。调用exit的应用程序将显示给用户崩溃,而不是执行正常终止并动画回到主屏幕。”
我认为这是一个不好的假设。如果用户点击退出按钮并显示一条消息,如下所示:“应用程序现在将退出。”,它似乎没有崩溃。 Apple应提供退出应用程序的有效方法(不是退出(0))。
这得到了一个很好的答案,但决定扩大一点:
如果不仔细阅读Apple的iOS人机界面指南,就无法将您的应用程序接受到AppStore。 (他们保留拒绝你对他们采取任何行动的权利)“不要以编程方式退出”http://developer.apple.com/library/ios/#DOCUMENTATION/UserExperience/Conceptual/MobileHIG/UEBestPractices/UEBestPractices.html这一部分是你在这种情况下应该如何对待的确切准则。
如果您遇到Apple平台问题,则无法轻松找到解决方案,请咨询HIG。 Apple可能根本不希望你这样做,而且他们通常(我不是Apple,所以我不能保证总是)在他们的文档中这样说。
我们不能使用exit(0)
,abort()
函数退出app,因为Apple强烈反对使用这些函数。虽然您可以将此功能用于开发或测试目的。
如果在开发或测试期间有必要终止应用程序,建议使用中止函数或断言宏
请找到这个Apple Q&A线程以获取更多信息。
使用此功能会产生应用程序崩溃的印象。所以我得到了一些建议,比如我们可以通过终止消息向知名用户显示关于关闭应用程序的警报,因为某些功能不可用。
但是Starting And Stopping App的iOS人机界面指南,建议永远不要使用Quit或Close按钮来终止Application。而不是他们建议显示正确的信息来解释情况。
iOS应用程序从不显示“关闭”或“退出”选项。当人们切换到另一个应用程序,返回主屏幕或将他们的设备置于睡眠模式时,他们会停止使用应用程序。
切勿以编程方式退出iOS应用。人们倾向于将此解释为崩溃。如果某些内容阻止您的应用按预期运行,您需要告诉用户有关情况并解释他们可以采取的措施。
除了以上,好的,我只想添加回答,考虑清理你的记忆。
应用程序退出后,iPhone OS将自动清除应用程序遗留的任何内容,因此手动释放所有内存只会增加应用程序退出所需的时间。
嗯,如果您的应用程序需要互联网连接,您可能“必须”退出该应用程序。您可以显示警报,然后执行以下操作:
if ([[UIApplication sharedApplication] respondsToSelector:@selector(terminate)]) {
[[UIApplication sharedApplication] performSelector:@selector(terminate)];
} else {
kill(getpid(), SIGINT);
}
- (IBAction)logOutButton:(id)sender
{
//show confirmation message to user
CustomAlert* alert = [[CustomAlert alloc] initWithTitle:@"Confirmation" message:@"Do you want to exit?" delegate:self cancelButtonTitle:@"Cancel" otherButtonTitles:@"OK", nil];
alert.style = AlertStyleWhite;
[alert setFontName:@"Helvetica" fontColor:[UIColor blackColor] fontShadowColor:[UIColor clearColor]];
[alert show];
}
- (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex
{
if (buttonIndex != 0) // 0 == the cancel button
{
//home button press programmatically
UIApplication *app = [UIApplication sharedApplication];
[app performSelector:@selector(suspend)];
//wait 2 seconds while app is going background
[NSThread sleepForTimeInterval:2.0];
//exit app when app is in background
NSLog(@"exit(0)");
exit(0);
}
}
我使用上面提到的[[NSMutableArray new] addObject:nil]方法强制退出(崩溃)应用程序而不进行tell-tale exit(0)函数调用。
为什么?因为我的应用程序在所有网络API调用上使用证书固定来防止中间人攻击。这些包括我的财务应用在启动时进行的初始化调用。
如果证书身份验证失败,我的所有初始化调用都会出错,并使我的应用程序处于不确定状态。让用户回家然后回到应用程序没有帮助,因为除非应用程序已被操作系统清除,否则它仍然未初始化且不值得信任。
因此,在这种情况下,我们认为最好弹出警报,通知用户应用程序在不安全的环境中运行,然后,当他们点击“关闭”时,强制使用上述方法退出应用程序。
[[UIApplication sharedApplication] terminateWithSuccess];
它工作正常并自动调用
- (void)applicationWillTerminateUIApplication *)application delegate.
删除编译时警告添加此代码
@interface UIApplication(MyExtras)
- (void)terminateWithSuccess;
@end
用户应决定何时退出应用。当应用程序退出时,我不认为这是一个很好的用户交互。因此它没有很好的API,只有主页按钮有一个。
如果出现错误:更好地实施或通知用户。如果必须重新启动:更好地实现它通知用户。
这听起来很愚蠢,但退出应用程序而不让用户决定而不通知他是不好的做法。 Apple表示,由于用户交互有一个主页按钮,因此同一功能(退出应用程序)不应该有两件事。
在iPhone上没有退出应用程序的概念。应该导致应用退出的唯一操作是触摸手机上的“主页”按钮,这不是开发人员可以访问的内容。
根据Apple的说法,您的应用不应该自行终止。由于用户未点击“主页”按钮,因此返回主屏幕会让用户感觉您的应用已崩溃。这是令人困惑的非标准行为,应该避免。
退出应用程序以外的方式除了主页按钮是非iOS风格的方法。
不过,我做了这个助手,没有使用私人内容:
void crash()
{ [[NSMutableArray new] addObject:NSStringFromClass(nil)]; }
但在我的情况下仍然不适合生产。它用于测试崩溃报告,或在核心数据重置后快速重启。如果函数留在生产代码中,那么就不会被拒绝。
可以使用名为Darvin
的库。
import Darwin
exit(0) // Here you go
NBC:iOS应用程序不推荐这样做。
执行此操作将获得崩溃日志。
你不应该直接调用函数exit(0)
,因为它会立即退出应用程序,看起来你的应用程序崩溃了。因此,最好向用户显示确认提醒并让他们自己执行此操作。
func askForQuit(_ completion:@escaping (_ canQuit: Bool) -> Void) {
let alert = UIAlertController(title: "Confirmation!", message: "Do you want to quit the application", preferredStyle: .alert)
alert.addAction(UIAlertAction(title: "Yes", style: UIAlertAction.Style.default, handler: { (action) in
alert.dismiss(animated: true, completion: nil)
completion(true)
}))
alert.addAction(UIAlertAction(title: "No", style: UIAlertAction.Style.cancel, handler: { (action) in
alert.dismiss(animated: true, completion: nil)
completion(false)
}))
self.present(alert, animated: true, completion: nil)
}
/// Will quit the application with animation
func quit() {
UIApplication.shared.perform(#selector(NSXPCConnection.suspend))
/// Sleep for a while to let the app goes in background
sleep(2)
exit(0)
}
self.askForQuit { (canQuit) in
if canQuit {
self.quit()
}
}
以其他方式退出应用程序
不过,我做了这个助手,没有使用私人内容:
出口(0);
退出应用程序可能是合适的,如果它是一个也在后台执行的长期应用程序,例如获取位置更新(使用位置更新后台功能)。
例如,假设用户退出基于位置的应用程序,并使用主页按钮将应用程序推送到后台。在这种情况下,您的应用可能会继续运行,但完全退出它可能是有意义的。这对用户来说是好事(释放不需要使用的内存和其他资源),并且有利于应用程序的稳定性(即确保应用程序在可能的情况下定期重新启动是针对内存泄漏和其他低内存的安全网问题)。
这可能(虽然可能不应该,见下文:-)通过以下方式实现:
- (void)applicationDidEnterBackground:(UIApplication *)application
{
if (/* logged out */) {
exit(0);
} else {
// normal handling.
}
}
由于应用程序将退出后台,因此用户看起来不会出错,并且不会像崩溃那样,只要用户界面在下次运行应用程序时就会恢复。换句话说,对于用户来说,当应用程序在后台时,它与应用程序的系统启动终止看起来没有任何不同。
尽管如此,最好使用更标准的方法让系统知道应用程序可以终止。例如,在这种情况下,通过停止请求位置更新来确保GPS未被使用,包括关闭地图视图上的当前位置(如果存在)。这样,系统将在应用程序进入后台后处理应用程序几分钟(即[[UIApplication sharedApplication] backgroundTimeRemaining]
)。这将获得所有相同的好处,而无需使用代码来终止应用程序。
- (void)applicationDidEnterBackground:(UIApplication *)application
{
if (/* logged out */) {
// stop requesting location updates if not already done so
// tidy up as app will soon be terminated (run a background task using beginBackgroundTaskWithExpirationHandler if needed).
} else {
// normal handling.
}
}
当然,使用exit(0)
永远不适合在前台运行的普通生产应用程序,因为其他答案参考http://developer.apple.com/iphone/library/qa/qa2008/qa1561.html
exit(0)对用户显示为崩溃,因此向用户显示确认消息。确认暂停后(主程按钮按编程方式按下)并等待2秒,当应用程序进入动画背景时,然后退出用户视图后面
-(IBAction)doExit
{
//show confirmation message to user
UIAlertView* alert = [[UIAlertView alloc] initWithTitle:@"Confirmation"
message:@"Do you want to exit?"
delegate:self
cancelButtonTitle:@"Cancel"
otherButtonTitles:@"OK", nil];
[alert show];
}
-(void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex
{
if (buttonIndex != 0) // 0 == the cancel button
{
//home button press programmatically
UIApplication *app = [UIApplication sharedApplication];
[app performSelector:@selector(suspend)];
//wait 2 seconds while app is going background
[NSThread sleepForTimeInterval:2.0];
//exit app when app is in background
exit(0);
}
}
它不是一种退出计划的方式,而是一种迫使人们戒烟的方法。
UIAlertView *anAlert = [[UIAlertView alloc] initWithTitle:@"Hit Home Button to Exit" message:@"Tell em why they're quiting" delegate:self cancelButtonTitle:nil otherButtonTitles:nil];
[anAlert show];
转到info.plist并选中“应用程序不在后台运行”键。这次当用户单击主页按钮时,应用程序完全退出。
在这里查看问答:https://developer.apple.com/library/content/qa/qa1561/_index.html
Q: How do I programmatically quit my iOS application?
没有为优雅地终止iOS应用程序提供API。
在iOS中,用户按下主页按钮以关闭应用程序。如果您的应用程序具有无法提供其预期功能的条件,建议的方法是为用户显示警告,指示问题的性质以及用户可能采取的操作 - 打开WiFi,启用位置服务等。允许用户自行决定终止申请。
警告:不要调用
exit
函数。调用exit
的应用程序将向用户显示已崩溃,而不是执行正常终止并动画回到主屏幕。此外,可能无法保存数据,因为如果调用exit,则不会调用
-applicationWillTerminate:
和类似的UIApplicationDelegate
方法。如果在开发或测试期间有必要终止您的应用程序,建议使用
abort
函数或assert
宏
在UIApplicationExitsOnSuspend
上添加application-info.plist
财产到true
。
经过一些测试,我可以说以下内容:
[UIApplication sharedApplication]
将导致应用程序看起来像崩溃,但它会在这之前调用- (void)applicationWillTerminate:(UIApplication *)application
;exit(0);
也将终止应用程序,但它看起来“正常”(跳板的图标显示为预期的,具有缩小效果),但它不会调用- (void)applicationWillTerminate:(UIApplication *)application
委托方法。我的建议:
- (void)applicationWillTerminate:(UIApplication *)application
。exit(0);
。您的ApplicationDelegate会收到用户有意退出的通知:
- (void)applicationWillResignActive:(UIApplication *)application {
当我收到此通知时,我只是打电话
exit(0);
所有的工作都做到了。最好的是,这是戒烟的意图,这就是为什么这不应该是在那里调用它的问题。
在我的音频应用程序上,当人们在音乐还在播放时同步他们的设备时,有必要退出应用程序。一旦同步完成,我会收到通知。但在此之后立即退出应用程序实际上看起来像是崩溃。
所以我设置了一个标志,以便在下一次后台操作时真正退出应用程序。可以在同步后刷新应用程序。