iOS 捕获Crash方法

  • iOS开发无法避免程序Crash问题,开发过程中可以用调试技术捕获Crash进行修复,但是发布到App Store后,无法抓到这些问题,因此目前国内友盟、听云等公司抓到Crash后将Crash时的堆栈上传到他们服务器上,并可以通过dSYM文件找到Crash的方法接口。

  • 当然我们自己也可以做到,首要问题是怎么抓到这些Crash原因和堆栈列表。iOS下面最常见的就是objective-c的NSException(通过@throw抛出,比如,NSArray访问元素越界、添加空对象等等)

  • Crash堆栈


    iOS Crash堆栈

设置方法

  • 1.导入头文件
#include <signal.h>
#include <execinfo.h>
  • 2.设置Crash捕获
- (void)initHandler {
    
    struct sigaction newSignalAction;
    memset(&newSignalAction, 0,sizeof(newSignalAction));
    newSignalAction.sa_handler = &signalHandler;
    sigaction(SIGABRT, &newSignalAction, NULL);
    sigaction(SIGILL, &newSignalAction, NULL);
    sigaction(SIGSEGV, &newSignalAction, NULL);
    sigaction(SIGFPE, &newSignalAction, NULL);
    sigaction(SIGBUS, &newSignalAction, NULL);
    sigaction(SIGPIPE, &newSignalAction, NULL);
    
    //异常时调用的函数
    NSSetUncaughtExceptionHandler(&handleExceptions);
}
  • 3.异常信息
void handleExceptions(NSException *exception) {
    NSLog(@"exception = %@",exception);
    NSLog(@"callStackSymbols = %@",[exception callStackSymbols]);
}

void signalHandler(int sig) {
    //最好不要写,可能会打印太多内容
    NSLog(@"signal = %d", sig);
}
  • 4.将异常信息同步到自己服务器上
    这样就可以抓到用户使用程序时的Crash了。

实例

  • 1.在应用程序启动时初始化设置捕获,在以下方法中调用initHandler方法进行设置。
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions```

- 2.写个异常程序验证,在某个控制器的```viewDidLoad```方法中写以下代码。

NSMutableArray *array = [NSMutableArray array];
[array addObject:nil];

- 3.运行代码

2016-07-10 15:44:01.148 iOSCrash[37234:6862176] exception = *** -[__NSArrayM insertObject:atIndex:]: object cannot be nil
2016-07-10 15:44:01.149 iOSCrash[37234:6862176] callStackSymbols = (
0 CoreFoundation 0x000000010ca89d85 __exceptionPreprocess + 165
1 libobjc.A.dylib 0x000000010c4fddeb objc_exception_throw + 48
2 CoreFoundation 0x000000010c94acc5 -[__NSArrayM insertObject:atIndex:] + 901
3 iOSCrash 0x000000010bffb80f -[ViewController viewDidLoad] + 111
4 UIKit 0x000000010cfda984 -[UIViewController loadViewIfRequired] + 1198
5 UIKit 0x000000010cfdacd3 -[UIViewController view] + 27
6 UIKit 0x000000010ceb0fb4 -[UIWindow addRootViewControllerViewIfPossible] + 61
7 UIKit 0x000000010ceb169d -[UIWindow _setHidden:forced:] + 282
8 UIKit 0x000000010cec3180 -[UIWindow makeKeyAndVisible] + 42
9 UIKit 0x000000010ce37ed9 -[UIApplication _callInitializationDelegatesForMainScene:transitionContext:] + 4131
10 UIKit 0x000000010ce3e568 -[UIApplication _runWithMainScene:transitionContext:completion:] + 1769
11 UIKit 0x000000010ce3b714 -[UIApplication workspaceDidEndTransaction:] + 188
12 FrontBoardServices 0x000000010f8a88c8 FBSSERIALQUEUE_IS_CALLING_OUT_TO_A_BLOCK + 24
13 FrontBoardServices 0x000000010f8a8741 -[FBSSerialQueue _performNext] + 178
14 FrontBoardServices 0x000000010f8a8aca -[FBSSerialQueue _performNextFromRunLoopSource] + 45
15 CoreFoundation 0x000000010c9af301 CFRUNLOOP_IS_CALLING_OUT_TO_A_SOURCE0_PERFORM_FUNCTION + 17
16 CoreFoundation 0x000000010c9a522c __CFRunLoopDoSources0 + 556
17 CoreFoundation 0x000000010c9a46e3 __CFRunLoopRun + 867
18 CoreFoundation 0x000000010c9a40f8 CFRunLoopRunSpecific + 488
19 UIKit 0x000000010ce3af21 -[UIApplication _run] + 402
20 UIKit 0x000000010ce3ff09 UIApplicationMain + 171
21 iOSCrash 0x000000010bffbb6f main + 111
22 libdyld.dylib 0x000000010f26392d start + 1
)

- 4.分析Crash

>4.1 从```exception```中得知Crash原因为Array中写了一个为nil的Object。
>4.2 从```callStackSymbols```堆栈符号中找到Crash的是第2条信息,堆栈列表可以理解为第0条为正在运行的,而该列表的第22是最早运行的,因此我们可以Crash的那条往下看可以知道这条Crash发生在```[ViewController viewDidLoad]```,即```ViewController```控制器的```viewDidLoad```方法中。

- 5.注意事项

>5.1 由于```NSSetUncaughtExceptionHandler```是```set```方法,所以只能设置一次,市面上很多抓Crash的库都会设置这个方法,因此,如果你用了UMeng Crash,Bugly(腾讯云内继承Bugly)等工具的话,最好不要自己设置,会影响Crash上报。

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,870评论 25 707
  • iOS开发中,解决Crash相信是开发者最为头疼的问题了,特别是对于已上线的应用,对其Crash的跟踪和修复显得尤...
    明仔Su阅读 7,572评论 7 77
  • 这篇接着iOS Crash处理方法(一):MethodSwizzle 继续描述Crash的问题。这篇跟上篇不同,上...
    晨寂阅读 4,016评论 15 11
  • 四.“诗人”和“小耗子” 都说“三个女子一台戏”,如今七个女孩同居一室,那可更是天天好戏连连看,精彩不错过。各位看...
    陈传容阅读 193评论 0 0
  • 接触了艺术这么久,却还是不能理解艺术界的一些奇葩事情。 梵高为什么要吃黄色颜料,戈雅的画作总是有一种奇妙的魔力,塞...
    Lavazza阅读 385评论 0 0