`

如何让IOS应用从容地崩溃

阅读更多

转自:http://blog.chukong-inc.com/index.php/2012/08/24/如何在ios应用中捕捉异常/

虽然大家都不愿意看到程序崩溃,但可能崩溃是每个应用必须面对的现实,既然崩溃已经发生,无法阻挡了,那我们就让它崩也崩得淡定点吧。

IOS SDK中提供了一个现成的函数 NSSetUncaughtExceptionHandler 用来做异常处理,但功能非常有限,而引起崩溃的大多数原因如:内存访问错误,重复释放等错误就无能为力了,因为这种错误它抛出的是Signal,所以必须要专门做Signal处理。步骤如下:

首先定义一个UncaughtExceptionHandler类,.h头文件的代码如下:

 

Source code    
#import <UIKit/UIKit.h>

@interface UncaughtExceptionHandler : NSObject
{
BOOL dismissed;
}
 
@end
 
void InstallUncaughtExceptionHandler();

 

然后在.mm文件实现InstallUncaughtExceptionHandler(),如下:

Source code    
void InstallUncaughtExceptionHandler()
{
signal(SIGABRT, MySignalHandler);
signal(SIGILL, MySignalHandler);
signal(SIGSEGV, MySignalHandler);
signal(SIGFPE, MySignalHandler);
signal(SIGBUS, MySignalHandler);
signal(SIGPIPE, MySignalHandler);
}

这样,当应用发生错误而产生上述Signal后,就将会进入我们自定义的回调函数MySignalHandler。为了得到崩溃时的现场信息,还可以加入一些获取CallTrace及设备信息的代码,.mm文件的完整代码如下:

#import “UncaughtExceptionHandler.h”
#include <libkern/OSAtomic.h>
#include <execinfo.h>

NSString * const UncaughtExceptionHandlerSignalExceptionName = @”UncaughtExceptionHandlerSignalExceptionName”;
NSString * const UncaughtExceptionHandlerSignalKey = @”UncaughtExceptionHandlerSignalKey”;
NSString * const UncaughtExceptionHandlerAddressesKey = @”UncaughtExceptionHandlerAddressesKey”;

volatile int32_t UncaughtExceptionCount = 0;
const int32_t UncaughtExceptionMaximum = 10;

const NSInteger UncaughtExceptionHandlerSkipAddressCount = 4;
const NSInteger UncaughtExceptionHandlerReportAddressCount = 5;

@implementation UncaughtExceptionHandler

+ (NSArray *)backtrace
{
void* callstack[128];
int frames = backtrace(callstack, 128);
char **strs = backtrace_symbols(callstack, frames);

int i;
NSMutableArray *backtrace = [NSMutableArray arrayWithCapacity:frames];
for (
i = UncaughtExceptionHandlerSkipAddressCount;
i < UncaughtExceptionHandlerSkipAddressCount +
UncaughtExceptionHandlerReportAddressCount;
i++)
{
[backtrace addObject:[NSString stringWithUTF8String:strs[i]]];
}
free(strs);

return backtrace;
}

- (void)alertView:(UIAlertView *)anAlertView clickedButtonAtIndex:(NSInteger)anIndex
{
if (anIndex == 0)
{
dismissed = YES;
}
}

- (void)handleException:(NSException *)exception
{
UIAlertView *alert =
[[[UIAlertView alloc]
initWithTitle:NSLocalizedString(@”Unhandled exception”, nil)
message:[NSString stringWithFormat:NSLocalizedString(
@"You can try to continue but the application may be unstable.\n"
@"%@\n%@", nil),
[exception reason],
[[exception userInfo] objectForKey:UncaughtExceptionHandlerAddressesKey]]
delegate:self
cancelButtonTitle:NSLocalizedString(@”Quit”, nil)
otherButtonTitles:NSLocalizedString(@”Continue”, nil), nil]
autorelease];
[alert show];

CFRunLoopRef runLoop = CFRunLoopGetCurrent();
CFArrayRef allModes = CFRunLoopCopyAllModes(runLoop);

while (!dismissed)
{
for (NSString *mode in (NSArray *)allModes)
{
CFRunLoopRunInMode((CFStringRef)mode, 0.001, false);
}
}

CFRelease(allModes);

NSSetUncaughtExceptionHandler(NULL);
signal(SIGABRT, SIG_DFL);
signal(SIGILL, SIG_DFL);
signal(SIGSEGV, SIG_DFL);
signal(SIGFPE, SIG_DFL);
signal(SIGBUS, SIG_DFL);
signal(SIGPIPE, SIG_DFL);

if ([[exception name] isEqual:UncaughtExceptionHandlerSignalExceptionName])
{
kill(getpid(), [[[exception userInfo] objectForKey:UncaughtExceptionHandlerSignalKey] intValue]);
}
else
{
[exception raise];
}
}

@end

NSString* getAppInfo()
{
NSString *appInfo = [NSString stringWithFormat:@"App : %@ %@(%@)\nDevice : %@\nOS Version : %@ %@\nUDID : %@\n",
[[NSBundle mainBundle] objectForInfoDictionaryKey:@”CFBundleDisplayName”],
[[NSBundle mainBundle] objectForInfoDictionaryKey:@”CFBundleShortVersionString”],
[[NSBundle mainBundle] objectForInfoDictionaryKey:@”CFBundleVersion”],
[UIDevice currentDevice].model,
[UIDevice currentDevice].systemName,
[UIDevice currentDevice].systemVersion,
[UIDevice currentDevice].uniqueIdentifier];
NSLog(@”Crash!!!! %@”, appInfo);
return appInfo;
}

void MySignalHandler(int signal)
{
int32_t exceptionCount = OSAtomicIncrement32(&UncaughtExceptionCount);
if (exceptionCount > UncaughtExceptionMaximum)
{
return;
}

NSMutableDictionary *userInfo =
[NSMutableDictionary
dictionaryWithObject:[NSNumber numberWithInt:signal]
forKey:UncaughtExceptionHandlerSignalKey];

NSArray *callStack = [UncaughtExceptionHandler backtrace];
[userInfo
setObject:callStack
forKey:UncaughtExceptionHandlerAddressesKey];

[[[[UncaughtExceptionHandler alloc] init] autorelease]
performSelectorOnMainThread:@selector(handleException:)
withObject:
[NSException
exceptionWithName:UncaughtExceptionHandlerSignalExceptionName
reason:
[NSString stringWithFormat:
NSLocalizedString(@"Signal %d was raised.\n"
@"%@", nil),
signal, getAppInfo()]
userInfo:
[NSDictionary
dictionaryWithObject:[NSNumber numberWithInt:signal]
forKey:UncaughtExceptionHandlerSignalKey]]
waitUntilDone:YES];
}

void InstallUncaughtExceptionHandler()
{
signal(SIGABRT, MySignalHandler);
signal(SIGILL, MySignalHandler);
signal(SIGSEGV, MySignalHandler);
signal(SIGFPE, MySignalHandler);
signal(SIGBUS, MySignalHandler);
signal(SIGPIPE, MySignalHandler);
}

在应用自身的 didFinishLaunchingWithOptions 前,加入一个函数:

Source code    
- (void)installUncaughtExceptionHandler
{
InstallUncaughtExceptionHandler();
}

最后,在 didFinishLaunchingWithOptions 中加入这一句代码就行了:

[self InstallUncaughtExceptionHandler];

现在,基本上所有崩溃都能Hold住了。崩溃时将会显示出如下的对话框:

这样在崩溃时还能从容地弹出对话框,比起闪退来,用户也不会觉得那么不爽。然后在下次启动时还可以通过邮件来发送Crash文件到邮箱,这就看各个应用的需求了。

分享到:
评论

相关推荐

    iOS简单版防崩溃处理

    "iOS简单版防崩溃处理"这一主题主要关注如何预防常见的编程错误导致的应用程序崩溃,如数组越界和非主线程更新用户界面(UI)。以下是对这些关键知识点的详细说明: 1. **数组越界**: 在Objective-C或Swift中,...

    用HTML5开发ios应用

    HTML5为iOS应用开发提供了一条捷径,让开发者可以使用他们熟悉的Web技术构建高性能、原生感的应用。随着技术的进步,HTML5在移动应用领域的应用越来越广泛,为开发者带来了更多的可能性和便利。

    ios应用入门指南

    ### iOS应用入门指南知识点梳理 #### 一、iOS应用开发概览 - **目标与价值**:iOS应用开发不仅能够带来乐趣,同时也是一项具有实际经济效益的活动。对于初学者而言,选择正确的起点至关重要。 - **必备条件**:...

    ios获取崩溃日志方法

    iOS 崩溃日志获取是 iOS 开发和测试中非常重要的一步,能够帮助开发者和测试人员快速定位崩溃原因,提高应用程序的稳定性和可靠性。下面将详细介绍获取 iOS 崩溃日志的方法。 什么是崩溃日志 崩溃日志是指应用程序...

    IOS 异常崩溃时发送邮件

    总之,通过`NSSetUncaughtExceptionHandler`和`MFMailComposeViewController`,开发者可以有效地收集并发送iOS应用的异常崩溃信息,这对于定位和修复问题至关重要。同时,了解如何优雅地处理异常和崩溃报告,也是...

    DSYM--iOS崩溃日志分析工具

    在iOS开发过程中,调试应用的崩溃日志是必不可少的工作,DSYM文件就是这个过程中至关重要的一个环节。DSYM,全称是DWARF with dSYM,它是一个包含了应用的调试信息的特殊文件,用于帮助开发者定位和分析iOS应用在...

    iOS 应用逆向工程 分析与实战

    dump、Theos、Cycript、IDA、GDB等最常用逆向工具的使用方法,进而分析使用这些工具时的思考方式和碰到问题时的解决思路,最后以社交应用消息拦截、iOS电话相关操作等4个极具代表性的实例总结iOS应用逆向工程在实战...

    ios应用程序编程指南

    开发者可以通过应用内浏览功能让应用用户在应用程序内部直接查看网页,这增强了应用程序的交互性和用户体验。 在iOS应用开发中,应用的安全性和兼容性也是非常重要的方面。例如,开发者需要了解如何通过URL方案实现...

    iOS开发 常见的崩溃闪退原因(12点)1

    总结,通过理解这些常见崩溃原因,开发者可以更好地预防和解决iOS应用中的问题。使用如腾讯Bugly或友盟等错误收集工具,可以帮助监控应用在真实环境中的表现,及时发现并修复问题,提高应用质量。

    iOS崩溃日志收集

    在iOS应用开发中,收集崩溃日志...总之,有效地收集和分析iOS崩溃日志是持续改进应用质量和用户体验的关键步骤。通过理解和运用上述知识点,开发者可以快速识别和解决应用中出现的各种异常,提升软件的稳定性和可靠性。

    iOS捕获程序异常崩溃记录、App信息等

    "iOS捕获程序异常崩溃记录、App信息等"这一主题就是关注如何在iOS应用中实现这样的功能。通过创建自定义的崩溃捕获机制,并将相关信息上传至服务器,我们可以获取详细的崩溃日志和系统信息,从而更好地诊断和解决...

    IOS7应用开发入门经典.第5版.pdf.zip

    《iOS7应用开发入门经典.第5版》是一本针对初学者深入浅出的iOS应用程序开发教程,专注于使用Objective-C编程语言。Objective-C是苹果公司为iOS和macOS平台开发应用的主要语言,它基于C语言并扩展了Smalltalk的面向...

    iOS防止崩溃

    通过这种方式,可以在发生异常时执行清理操作,或者至少向用户显示友好的错误信息,而不是让应用直接崩溃。 2. **使用异常断点**:Xcode提供了一种称为“异常断点”的功能,当应用抛出异常时,它会暂停执行。这有助...

    高仿微信,iOS应用开发模板.zip ios 开发模板

    高仿微信,iOS应用开发模板.zip ios 开发模板。高仿微信,iOS应用开发模板.zip ios 开发模板。高仿微信,iOS应用开发模板.zip ios 开发模板。高仿微信,iOS应用开发模板.zip ios 开发模板。高仿微信,iOS应用开发...

    马上着手开发iOS应用程序(苹果官方)

    《马上着手开发iOS应用程序》是苹果官方发布的一份教程,旨在帮助初学者快速入门iOS应用开发。这份教程在2014年底更新至最新版本,但后来已被苹果从其官方平台下架,因此显得尤为珍贵。它涵盖了iOS开发的基础知识和...

    Swift iOS应用开发实战49955.zip

    Swift是苹果公司推出的一种强大的编程语言,主要用于iOS、macOS、watchOS和tvOS的应用开发。这个名为"Swift iOS应用开发实战49955.zip"的资源包含了使用Swift进行iOS应用开发的实战教程和源码,很可能是为了帮助...

    iOS异常捕获和崩溃日志

    当iOS应用遇到未预见的错误或条件时,它可能会突然崩溃,给用户带来不愉快的体验。为了调试和优化这些情况,开发者需要有效地捕获异常并记录详细的崩溃日志。 首先,我们要理解什么是异常。在Objective-C中,异常是...

Global site tag (gtag.js) - Google Analytics