iOS - 崩溃异常处理(1)

https://www.jianshu.com/p/4d32664dcfdb

一、关于崩溃

闪退估计是我们最不想看到的,对于用户而言,马上就能产生一种不悦,对于投资方而言,也会产生对技术实力的不信任感,所以,我们就需要对闪退进行处理,这里介绍一个不错的三方:AvoidCrash,写这个的大大也很牛逼,原文参照这里

这个三方可以处理例如插入空值到字典中或数组中引起的崩溃、数组越界引起的崩溃、unrecognized selector sent to instance等等的崩溃,都能捕获并且避免闪退。

对于插入空值、越界等,原理比较简单,就是利用Runtime的方法交换,把普通的插入和取值的方法,替换成安全插入和安全读取的方法,具体代码可以去看源码。
话不多说,先上效果:
以下是可导致崩溃的代码:

    NSString *nilStr = nil;
    NSArray *array = @[@"chenfanfang", nilStr];

崩溃截图

若有AvoidCrash来防止崩溃,则不会崩溃,并且会将原本会崩溃情况的详细信息打印出来,如下图:

防止崩溃的效果

效果不错吧,接下来上使用步骤:

  • 集成:
    建议使用cocoapod,仅需要pod AvoidCrash一句话即可。(手动导入的步骤,可以参照上面所说的原文)。
  • 使用方法:(只要在AppDelegatedidFinishLaunchingWithOptions方法中调用avoidCrash方法,就可以开始监听异常。)
- (void)avoidCrash {

    /*
     * 项目初期不需要对"unrecognized selector sent to instance"错误进行处理,因为还没有相关的崩溃的类
     * 后期出现后,再使用makeAllEffective方法,把所有对应崩溃的类添加到数组中,避免崩溃
     * 对于正式线可以启用该方法,测试线建议关闭该方法
     */
    [AvoidCrash becomeEffective];

//    [AvoidCrash makeAllEffective];
//    NSArray *noneSelClassStrings = @[
//                                     @"NSString"
//                                     ];
//    [AvoidCrash setupNoneSelClassStringsArr:noneSelClassStrings];

    //监听通知:AvoidCrashNotification, 获取AvoidCrash捕获的崩溃日志的详细信息
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(dealwithCrashMessage:) name:AvoidCrashNotification object:nil];
}
  • 再监听异常的通知:
- (void)dealwithCrashMessage:(NSNotification *)notification {
    MYLog(@"\n??\n??监测到崩溃信息??\n??\n");
    /*
     * 在这边对避免的异常进行一些处理,比如上传到日志服务器等。
     */
}
以上就是避免崩溃的简单用法,关于能处理哪些异常,可以自行查看Git中的项目介绍。

二、关于异常的统计

上述的方法,能够避免崩溃,但是不能够避免所有状况的崩溃,作者也在不断的根据用户的使用情况进行更新,尽量对所有已知的崩溃进行避免。所以,我们还需要对异常进行其他的收集,也能有效的帮助自己改进APP。

这里仅做腾讯的Bugly进行介绍,因为其他的例如友盟、极光的,个人感觉都没有Bugly好用,我也就不做介绍了,有兴趣的可以自行了解。
这里参照的文章原文在此

  • 集成
    集成很简单,按照官方文档来就好,我们这里建个简单的小项目,模拟一些崩溃,测试下Bugly的bug上报及时性。
    项目就取名叫NSException了,创建好项目后,去Bugly的控制台,添加我们的应用。

    添加应用

    创建完应用,进入下一个界面,我们选择异常上报。

    选择异常上报

    再到我们的APP的appDelegatedidFinishLaunchingWithOptions方法内调用初始化方法即可。

// 头文件
#import <Bugly/Bugly.h>

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {

    [Bugly startWithAppId:@"此处替换为你的AppId"];

    return YES;
}

AppID可以点击你在控制台创建的App,然后点产品设置就能看到了。

查看appid

  • Bug上传测试
    接下来我们在ViewConroller中随便创造一个闪退的bug
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
    NSArray *arr = @[@"", @""];
    arr[5];
}

运行,崩溃,刷新Bugly的控制台,你会发现,bug已经统计到了。所以,Bugly的崩溃上传是在崩溃后立刻上传的。

控制台记录的崩溃信息

我们点进异常问题中去看一下,崩溃信息大致是这样的,我们可以很直观的看到崩在哪个方法里了。

异常信息

进阶

如果我们就这样使用Bugly是不是太可惜了,我们来看看Bugly还有什么功能;查看头文件,会发现Bugly有三个类暴露出来,分别是BuglyBuglyConfigBuglyLog

1.BuglyConfig类主要用于个性话配置Bugly类,由一些属性和BuglyDelegate代理组成。
  • 属性:BuglyConfig大部分属性有设有默认值,一般不用更改,但是关于卡顿监控的属性确是默认关闭的:
/**
*  卡顿监控开关,默认关闭
*/
@property (nonatomic) BOOL blockMonitorEnable;
/**
*  卡顿监控判断间隔,单位为秒
*/
@property (nonatomic) NSTimeInterval blockMonitorTimeout;

如果需要上报卡顿,只需要将blockMonitorEnable设为true,给blockMonitorTimeout设置一个合理的值即可;

  • 代理:BuglyConfig可以设置一个代理,来自定义上传崩溃的附属信息;
@protocol BuglyDelegate <NSObject>

@optional
/**
*  发生异常时回调
*  @param exception 异常信息
*  @return 返回需上报记录,随异常上报一起上报
*/
- (NSString * BLY_NULLABLE)attachmentForException:(NSException * BLY_NULLABLE)exception;
@end

我们的初始化就改成:

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
    BuglyConfig *config = [[BuglyConfig alloc] init];
    //监听卡顿
    config.blockMonitorEnable = YES;
    config.blockMonitorTimeout = 3;
    config.consolelogEnable = YES;
    config.delegate = self;
    [Bugly startWithAppId:@"此处替换为你的AppId" config:config];
//    [self avoidCrash];
    return YES;
}
- (NSString *)attachmentForException:(NSException *)exception {
    NSLog(@"异常事件代理");
    return [NSString stringWithFormat:@"TEST: %@",exception.userInfo];
}

再运行一次,崩溃,然后我们看看Bugly控制台上报记录:

留意红色边框内的文件

文件内部记录的就是异常的代理方法所上报的内容。

2.上传打印日志,BuglyLog类主要用于打印日志,有6种级别:
typedef NS_ENUM(NSUInteger, BuglyLogLevel) {
  BuglyLogLevelSilent  = 0,
  BuglyLogLevelError   = 1,
  BuglyLogLevelWarn    = 2,
  BuglyLogLevelInfo    = 3,
  BuglyLogLevelDebug   = 4,
  BuglyLogLevelVerbose = 5,
};

BuglyLog除了控制台打印,还有一个重要功能就是上报打印内容,内容将在崩溃时一同被上报;但是这个功能是默认不上报的,需要配置BuglyConfig的reportLogLevel属性;如config.reportLogLevel = BuglyLogLevelWarn,将会上报BuglyLogLevelWarn和BuglyLogLevelError级别的打印日志。

3.自定义上报异常,我们再回到Bugly类,除了初始化Bugly的方法外,还有一些其他的方法:

自定义上报错误

/**
*  上报自定义异常
*  @param exception 异常信息
*/
+ (void)reportException:(nonnull NSException *)exception;
/**
*  上报错误
*  @param error 错误信息
*/
+ (void)reportError:(NSError *)error;

重点来了!!!!

配合上AvoidCrash,使用上报自定义异常方法,我们就既能避免崩溃,又能监听异常!

//AvoidCrash异常通知监听方法,在这里我们可以调用reportException方法进行上报
- (void)dealwithCrashMessage:(NSNotification *)notification {
    NSLog(@"\n??\n??监测到崩溃信息??\n??\n");

    NSException *exception = [NSException exceptionWithName:@"AvoidCrash" reason:[notification valueForKeyPath:@"userInfo.errorName"] userInfo:notification.userInfo];
    [Bugly reportException:exception];
}

以上就是AvoidCrash+Bugly优化APP的运行处理。

虽然Bugly的崩溃列表中我们能看到得到代码的崩溃信息,但想更具体的分析代码位置,就要用到符号表了。

三、符号表

没有符号表,我们就无法定位崩溃中的符号对应的代码所在的类以及类中的行数位置。我们在每次构建版本、debug的时候,都会生成dSYM后缀名的符号表文件,而我们App在手机上运行的时候,崩溃后产生的崩溃信息,不可能定位到代码的多少多少行,因为这些信息对于App运行是没有意义的,存储在App中势必会增大安装包的体积,所以App的崩溃信息都是存储为各种符号,具体符号代表什么,需要去符号表中查找对应的含义。
我们每次debug、构建版本,都会生成dSYM文件,都对应了一个UUID(像我们的手机一样,都有一个唯一标志),按下图指示,我们就能找到我们所使用的App版本对应的dSYM文件的UUID,通过这个UUID,我们就能找到存储在我们电脑中的dSYM文件,将这个文件上传到bugly,bugly会自动帮我们找到崩溃符号的含义。

查看符号表文件

需要注意的是,构建版本会自动生成dSYM文件,但debug的时候,是没有的,需要我们手动开启。在build setting中搜索debug,将下面两项内容修改为正确的设置:

手动开始debug模式下的生成dSYM文件

有了符号表的UUID,我们打开终端,按UUID找到符号表的路径。
mdfind "com_apple_xcode_dsym_uuids == A8E87810-70A7-3335-B638-C8B01BE15D79"
后面的一串字母数字组合,就是我们的UUID,这里需要将UUID按一定格式处理下,也就是在特定位置插入“-”,具体格式如下:

处理UUID

来到终端,运行上面的命令,就定位到了dSYM文件的位置:

定位dSYM文件

打开文件路径,就找到了dSYM文件:

找到dSYM文件

拷贝出来,压缩为zip文件,上传到bugly上。

上传文件

刷新页面,再回去看刚才的问题,定位到了为ViewController.m的第24行:

重新查看异常

这样我们就定位到了有问题的地方。

官网文档也提供了自动上传dSYM文件的操作流程,有兴趣的可以试试,避免以后每次新版本都要手动上传dSYM文件。

dSYM文件也可以手动查找:

手动查找dSYM文件

找到你构建的版本,右键show in finder:

查看finder中路径

然后在定位到的文件上右键显示包内容就OK了:

包内查找dSYM文件

原文地址:https://www.cnblogs.com/baitongtong/p/10954322.html

时间: 2024-08-04 09:51:02

iOS - 崩溃异常处理(1)的相关文章

IOS崩溃日志解析(crash log)

IOS的应用程序少不了crash,互联网统计分析工具友盟有一项目错误分析的功能,专门用于应用程序崩溃日志统计,最近研究友盟上统计到的崩溃日志,在此对崩溃日志做一个简单的总结. IOS崩溃日志分类: 一.低内存崩溃:IOS设备检测到低内存时,虚拟内存系统发出通知请求应用释放内存.这些通知发送到所有正在运行的应用和进程,试图收回一些内存.如果内存使用依然居高不下,系统将会终止后台线程以缓解内存压力.如果可用内存足够,应用将能够继续运行而不会产生崩溃报告.否则,应用将被iOS终止,并产生低内存崩溃报告

iOS崩溃日志

今天看crash report ,有这样两个crash: 调用 stopUpdatingLocation 函数的是一个CLLocationManager 类型的对象,为什么报错的时候会把这个对象转成NSConcerteAttributedString类型? 调用getEsubmissionResult方法的是一个UIViewController类型的对象,为什么会转成_NSCFString类型? iOS崩溃日志 >> ios 这个答案描述的挺清楚的:http://www.goodpm.net/

转: iOS崩溃堆栈符号表使用与用途

转:http://bugly.qq.com/blog/?p=119 iOS崩溃堆栈符号化,定位问题分分钟搞定! 2015.3.16 腾讯Bugly 微信分享 最近一段时间,在跟开发者沟通过程中,萝莉发觉大家对iOS的应用符号表还不是很清楚,除了咨询关于符号表生成.配置的问题以外,对Bugly崩溃分析需要配置符号表也存在疑问. 在这里,萝莉就给大家分享下关于iOS符号表的一些内容. 首先,进行常识“脑补”. 1. 符号表是什么? 符号表就是指在Xcode项目编译后,在编译生成的二进制文件.app的

ios 崩溃日志揭秘

http://www.raywenderlich.com/zh-hans/30818/ios%E5%BA%94%E7%94%A8%E5%B4%A9%E6%BA%83%E6%97%A5%E5%BF%97%E6%8F%AD%E7%A7%98 0x8badf00d: 读做 "ate bad food"! (把数字换成字母,是不是很像 :p)该编码表示应用是因为发生watchdog超时而被iOS终止的.  通常是应用花费太多时间而无法启动.终止或响应用系统事件. 0xbad22222: 该编码

iOS崩溃堆栈信息的符号化解析

最近一段时间,在iOS开发调试过程中以及上线之后,程序经常会出现崩溃的问题.简单的崩溃还好说,复杂的崩溃就需要我们通过解析Crash文件来分析了,解析Crash文件在iOS开发中是比较常见的.但在跟开发者沟通过程中,云捕小编发觉大家对iOS的应用符号表还不是很清楚. 现在网上有很多关于解析崩溃堆栈信息的符号化的博客,但是大多质量参差不齐,或者有些细节没有注意到.今天总结一下对iOS崩溃符号化的使用和技巧: 一.场景 当我们收集iOS的崩溃信息时,获取到的崩溃堆栈一般是如下的形式,全是十六进制的内

iOS之异常处理

每当出现bug或者crash的时候,我们总是习惯性的加入了NSLog或则单步调试.但是往往有些时候总找不到莫名奇妙的原因,回头看看C++对于异常处理的一些总结. @try {     <#Code that can potentially throw an exception#> } @catch (NSException *exception) {     <#Handle an exception thrown in the @try block#> } @finally {

常用获取Android崩溃日志和IOS崩溃日志的几种方法

一:前言 在日常测试app时,经常会遇到崩溃问题,测试快速抓取到崩溃日志可以有效方便开发进行定位,快速解决问题所在测试做到测试分析,定位是非常重要的,这也是判断一个测试能力指标的一大维度. 二:Android崩溃日志 一.通过adb logcat获取 # 清除日志,新手上路时,日志内容很多,对于能毕现的日志,可以先清除后重新获取 adb logcat -c # 然后再次运行崩溃操作,再抓取日志 # 存储日志到当前目录下的 carsh.log 中 adb logcat -d *:W > crash

iOS崩溃前日志记录实现

如何使用Signal 因为错误抛出Signal,必须要专门做Signal处理. 在计算机科学中,信号(英语:Signals)是Unix.类Unix以及其他POSIX兼容的操作系统中进程间通讯的一种有限制的方式.它是一种异步的通知机制,用来提醒进程一个事件已经发生.当一个信号发送给一个进程,操作系统中断了进程正常的控制流程,此时,任何非原子操作都将被中断.如果进程定义了信号的处理函数,那么它将被执行,否则就执行默认的处理函数. 在项目工程中,要使用 #include <sys/signal.h>

iOS崩溃调试的使用和技巧总结

每日更新关注:http://weibo.com/hanjunqiang  新浪微博 在iOS开发调试过程中以及上线之后,程序经常会出现崩溃的问题.简单的崩溃还好说,复杂的崩溃就需要我们通过解析Crash文件来分析了,解析Crash文件在iOS开发中是比较常见的. 现在网上有很多关于解析崩溃信息的博客,但是大多质量参差不齐,或者有些细节没有注意到.今天写一篇博客总结一下我对崩溃调试的使用和技巧,如果有哪些错误或遗漏,还请指点,谢谢! 获取崩溃信息 在iOS中获取崩溃信息的方式有很多,比较常见的是使