IOS崩溃 异常处理(NSSetUncaughtExceptionHandler)

iOS已发布应用中对异常信息捕获和处理
代码下载地址:http://download.csdn.net/detail/daiyelang/6740205

iOS开发中我们会遇到程序抛出异常退出的情况,如果是在调试的过程中,异常的信息是一目了然,但是如果是在已经发布的程序中,获取异常的信息有时候是比较困难的。

iOS提供了异常发生的处理API,我们在程序启动的时候可以添加这样的Handler,这样的程序发生异常的时候就可以对这一部分的信息进行必要的处理,适时的反馈给开发者。

不足的地方是,并不是所有的程序崩溃都是由于发生可以捕捉的异常的,有些时候是因为内存等一些其他的错误导致程序的崩溃,这样的信息是不在这里体现的。

我做了一个简单的类,进行很基本的操作,可以添加和获取Handler,捕获到异常后将信息写入到app的Documens下的Exception.txt中。

其实还有很多的处理的办法。
l 比如可以在程序下一次起来的时候读取这个异常文件发生到服务端。
l 或者直接就是在处理代码中用openurl的方式(mailto:)调用发送邮件的方式,将异常信息直接变成邮件发送到指定地址。

以下是完整的代码实现。

使用场景示例:

#pragma mark -  
#pragma mark Application lifecycle  
   
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {     
     
    // Override point for customization after application launch.  
      
    [window makeKeyAndVisible];  
     [NdUncaughtExceptionHandler setDefaultHandler];  
     NSArray *array = [NSArray arrayWithObject:@"there is only one objective in this arary,call index one, app will crash and throw an exception!"];  
     NSLog(@"%@", [array objectAtIndex:1]);  
      
     return YES;  
}  

基本接口展示:

#import <Foundation/Foundation.h>  

@interface NdUncaughtExceptionHandler : NSObject {  

}  

+ (void)setDefaultHandler;  
+ (NSUncaughtExceptionHandler*)getHandler;  

@end  
//还可以选择设置自定义的handler,让用户取选择  

接口实现展示

#import "NdUncaughtExceptionHandler.h"  
   
NSString *applicationDocumentsDirectory() {  
    return [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject];  
}  
   
void UncaughtExceptionHandler(NSException *exception) {  
     NSArray *arr = [exception callStackSymbols];  
     NSString *reason = [exception reason];  
     NSString *name = [exception name];  
   
     NSString *url = [NSString stringWithFormat:@"=============异常崩溃报告=============\nname:\n%@\nreason:\n%@\ncallStackSymbols:\n%@",  
                   name,reason,[arr componentsJoinedByString:@"\n"]];  
     NSString *path = [applicationDocumentsDirectory() stringByAppendingPathComponent:@"Exception.txt"];  
     [url writeToFile:path atomically:YES encoding:NSUTF8StringEncoding error:nil];  
     //除了可以选择写到应用下的某个文件,通过后续处理将信息发送到服务器等  
     //还可以选择调用发送邮件的的程序,发送信息到指定的邮件地址  
     //或者调用某个处理程序来处理这个信息  
}  
   
@implementation NdUncaughtExceptionHandler  
   
-(NSString *)applicationDocumentsDirectory {  
    return [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject];  
}  
   
+ (void)setDefaultHandler  
{  
     NSSetUncaughtExceptionHandler (&UncaughtExceptionHandler);  
}  
   
+ (NSUncaughtExceptionHandler*)getHandler  
{  
     return NSGetUncaughtExceptionHandler();  
}  
   
@end  

异常崩溃报告:
=============异常崩溃报告=============

name:  
NSRangeException  
reason:  
*** -[NSArray objectAtIndex:]: index 1 beyond bounds [0 .. 0]  
callStackSymbols:  
0   CoreFoundation                      0x02393919 __exceptionPreprocess + 185  
1   libobjc.A.dylib                     0x024e15de objc_exception_throw + 47  
2   CoreFoundation                      0x0238958c -[__NSArrayI objectAtIndex:] + 236  
3   UncaughtE                           0x000022e8 -[UncaughtEAppDelegate application:didFinishLaunchingWithOptions:] + 157  
4   UIKit                               0x002b8543 -[UIApplication _callInitializationDelegatesForURL:payload:suspended:] + 1163  
5   UIKit                               0x002ba9a1 -[UIApplication _runWithURL:payload:launchOrientation:statusBarStyle:statusBarHidden:] + 346  
6   UIKit                               0x002c4452 -[UIApplication handleEvent:withNewEvent:] + 1958  
7   UIKit                               0x002bd074 -[UIApplication sendEvent:] + 71  
8   UIKit                               0x002c1ac4 _UIApplicationHandleEvent + 7495  
9   GraphicsServices                    0x02bf9afa PurpleEventCallback + 1578  
10  CoreFoundation                      0x02374dc4 __CFRUNLOOP_IS_CALLING_OUT_TO_A_SOURCE1_PERFORM_FUNCTION__ + 52  
11  CoreFoundation                      0x022d5737 __CFRunLoopDoSource1 + 215  
12  CoreFoundation                      0x022d29c3 __CFRunLoopRun + 979  
13  CoreFoundation                      0x022d2280 CFRunLoopRunSpecific + 208  
14  CoreFoundation                      0x022d21a1 CFRunLoopRunInMode + 97  
15  UIKit                               0x002ba226 -[UIApplication _run] + 625  
16  UIKit                               0x002c5b58 UIApplicationMain + 1160  
17  UncaughtE                           0x00002228 main + 102  
18  UncaughtE                           0x000021b9 start + 53  

不足的地方是,并不是所有的程序崩溃都是由于发生可以捕捉的异常的,有些时候引起崩溃的大多数原因如:内存访问错误,重复释放等错误就无能为力了,因为这种错误它抛出的是Signal,所以必须要专门做Signal处理。首先定义一个UncaughtExceptionHandler类,.h头文件的代码如下:

UncaughtExceptionHandler类,.h头文件的代码如下:

#import  
 
@interface UncaughtExceptionHandler : NSObject{  
BOOL dismissed;  
}  
@end  

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

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 #include  
 
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);  
 
   charchar **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 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])  
 
  {  
<span style="white-space:pre"> </span>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 前,加入一个函数:

- (void)installUncaughtExceptionHandler  
{  
InstallUncaughtExceptionHandler();  
}  

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

[self InstallUncaughtExceptionHandler];  

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

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

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,482评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,377评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,762评论 0 342
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,273评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,289评论 5 373
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,046评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,351评论 3 400
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,988评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,476评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,948评论 2 324
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,064评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,712评论 4 323
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,261评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,264评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,486评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,511评论 2 354
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,802评论 2 345

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,599评论 18 139
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,581评论 18 399
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,510评论 25 707
  • 最近回老家参加面试考试,画画上就懈怠了,昨晚一气呵成画了四幅,画到凌晨一点,老公在外地连夜火车赶回来看我 和宝宝,...
    天然呆自然傻阅读 542评论 5 10
  • 人是最没记性的动物,经常会流连忘返那些伤过自己的人,明知道这个人最后和自己没关系,但仍然想试试,究竟结果如何,永远...
    茶二少爷阅读 520评论 0 0