iOS 闪退收集

iOS 闪退收集_第1张图片

一、dSYM

什么是dSYM

dSYM实际上放的是程序在编译过程中收集的符号表的信息,其实质上保存符号表数据的是二进制的DWARF(DebuggingWith Arbitrary Record Formats)文件。可以用dwarfdump命令读取出一些可读的信息。

设置Xcode工程生成dSYM

  • Build Setting中设置Debug Information Format 为 DWARF with dSYM File
  • Build Setting中设置Generate Debug Symbols 为 Yes

  • 小技巧:当本地调试不需要dSYM的时候,可以设置XCode不生成dSYM,这样做的好处是使得XCode编译运行的速度更快(少了生成dSYM的时间)。

查看dSYM的UUID信息


这个步骤很关键,这是确定dSYM是否可用的唯一标识,用来与崩溃日志进行配对。
注意:
1. 只要代码是相同的,编译后产生的dSYM都有一样的UUID,也就意味着你的应用出现闪退,而dSYM丢失的情况,只要工程还在,代码不变就可以重新编译生成dSYM,用来解析崩溃日志。
2. dSYM的目录可以重命名,但是.dSYM的后缀不能去掉,不能放在.dSYM的目录文件夹中。

二、崩溃日志

移动设备产生崩溃日志的情况

  • 应用违反操作系统的规则,主要包括三种情况,watchdog超时、用户强制退出和低内存终止。
  • 应用中有bug,这是我们应该关心的情况。

崩溃日志的获取途径(很关键)

  • 手机端:不同的iOS版本在不同的目录下,我的iOS11.2在 设置->隐私->分析->分析数据,可以看到很多不同的.ips文件。
  • 电脑端:使用iTools的工具,选择工具箱栏目,有一个崩溃日志的选项,文件名的格式是”应用名-闪退时间.ips“。
  • 闪退收集框架:如果在应用中接入了闪退收集的框架,或者自己实现了闪退收集的模块,那就还可以通过这些模块获取闪退的日志信息。

了解崩溃日志的文件结构

* Incident Identifier: 崩溃日志的唯一标识符,对应不同的crash
* CrashReporter Key:   设备的标识key值,不同于设备的UDID,当我们拿到大量的闪退日志而只有几个CrashReport Key时可能意味着闪退值发生在个别机器。    
* Hardware Model:      设备类型 
* Process:             进程名,一般就是我们的AppName,[]中的数字表示的是进程ID
* Path:                可执行程序存储在设备的路径名
* Identifier:          App的Indentifier,来自于你的证书
* Version:             App的版本,info.plist中CFBundleShortVersionString和CFBundleVersion这两个字段
* Code Type:           当前CPU架构
* Role:                异常发生时该进程task_role的值,貌似是描述进程的结构体某个成员变量的枚举值
* Parent Process:      父进程
* Report Version:      崩溃日志的版本,目前就见过104和105
* Exception Type:      异常的类型  
* Thread State:        线程回溯
* Binary Images:       这段内容列出了在线程终止的时候,进程的二进制映像  

符号化解析的过程(可不必过多纠结)

详细过程可以参见/Applications/Xcode.app/Contents/SharedFrameworks/DVTFoundation.framework/Versions/A/Resources/symbolicatecrash,这实际上就是一份perl脚本。

  1. 使用符号化解析工具,第一步是指定脚本中$DEVELOPER_DIR的值,使用如下命令
    export DEVELOPER_DIR="/Applications/Xcode.app/Contents/Developer"
  1. 进行符号化解析
    执行如下命令,实际测试可以发现symbolicatecrash工具最少只需要输入闪退日志就可以进行解析,下面这条命令会把crash.ips解析结果导出到out.log,-v选项可以把解析过程中的一些日志信息打印到标准出错。另外因为symbolicatecrash有个bug,当闪退日志中相邻的重复行太多的时候,命令会卡死,所以在进行解析前先使用uniq去重,再通过管道流向symbolicatecrash即可。
    uniq crash.ips | symbolicatecrash -v > out.log

其他的一些选项可以查看symbolicatecrash的详细帮助信息。

    usage: 
    symbolicatecrash [--help] [--dsym=DSYM] [--output OUTPUT_FILE]  [SYMBOL_PATH ...]

    <LOGFILE>                   The crash log to be symbolicated. If "-", then the log will be read from stdin
    <SYMBOL_PATH>               Additional search paths in which to search for symbol rich binaries
    -o | --output  The symbolicated log will be written to OUTPUT_FILE. Defaults to "-" (i.e. stdout) if not specified
    -d | --dsym    Adds additional dSYM that will be consulted if and when a binary's UUID matches (may be specified more than once)
    -h | --help                 Display this help message
    -v | --verbose              Enables additional output

上述解析命令能执行成功的前提条件是解析命令的本机是有对应的dSYM存在的。
此外还有一个实际应用中的问题需要补充,因为我们的解析有可能是后台的一个服务,所以解析的过程可能在一个脚本中执行,而脚本解析闪退日志的第一步是下载服务器上的dSYM,一般会是一个压缩包,然后进行解压和符号化解析,这时候解析的过程很可能会失败。
猜测原因是symbolicatecrash工具使用mdfind工具在Mac上进行全局搜索,而由于Mac的缓存机制,刚解压的dSYM,可能还不被认为是dSYM(也就是说缓存机制在一定时间之后会把这个解压出来的dSYM加入到mdfind的搜索范围中,在这之前mdfind找不到),所以会导致闪退日志找不到解析不出来。解决办法是使用mdimport强制刷新刚解压的dSYM的目录,再进行符号化解析就没问题了。

三、如何解析闪退日志,看懂闪退日志(敲黑板划重点)

分析闪退日志

  • 首先找到闪退发生的闪退类型,如”EXC_BAD_ACCESS”,拿着这个闪退类型去到Apple的官网,可以确定一个大致的方向。
  • 如果是直接从手机拿的系统的闪退日志,异常子类型的字段(Exception Codes)可能会有一串特殊含义的编码(英文或数字都有可能),可以去Apple官网或者百度搜索其具体含义,以获得进一步的异常信息。
  • 在闪退日志中可能会有这样的字段”Application Specific Information:”,这是闪退日志中最有用的一段信息,这段信息意味着某个NSException导致程序Crash,且这段信息包含了NSException的reason和name,同时还会有”Last Exception Backtrace:”的一段信息,可以精准的定位到程序出错的位置。
  • 找到闪退对应的线程,大部分情况是线程0闪退,分析闪退线程对应的堆栈,找到自己应用的应用名对应的行,一般会解析出函数名、行号和文件名,这里大概就是问题的所在。

闪退日志解析不完整的原因

有以下两种情况(可能还有未知的原因)

* 系统的库解不出来, 是因为~/Library/Developer/Xcode/iOS\ DeviceSupport目录下缺失对应的版本系统库的dSYM,可以把对应的设备插在Mac机上,可以自动生成这些文件。
* 用户库解析不出来, 这是因为用户库的dSYM不完整,可能是如下的情况,游戏或者SDK有引用到动态库,而动态库的dSYM是需要额外提供的,所以会解析不出来。

调试的过程中获取闪退日志

在本地打包调试的时候,也可以直接使用Xcode查看闪退日志,这种方式更加的方便,可以看到xcode自动会把闪退日志进行解析,不过前提是XCode已经设置生成dSYM了。

四、简单说下闪退解析服务的完整解决方案

app层面:

  1. 接入第三方的闪退收集框架(可以自己实现,原理不难),开源的如:PLCrashReport。
  2. 每次进入app之前检查上一次打开应用是不是出现了闪退,如果是则把上一次出现的闪退日志进行上报,上报的时候可以把用户信息,设备信息,或者玩家等级等信息一并上传,方便进行统计分析。

后台服务:

  1. 接收app层的闪退上报。
  2. ios打包的时候应该把ipa跟dSYM对应的管理起来(亲测在 product name 设置标记是可行的),解析之前去服务器上下载闪退日志对应的dSYM。
  3. 按照上述的方式进行闪退日志的解析,另外如果有需求的话,可以对上报的内容进行分类统计(前面有说到,app层面可以把用户信息上传)。
  4. 闪退日志上传之后最好按照闪退的类型进行分类,分类的方法(这里我是参考的腾讯Bugly):首先按包分类,其次如果闪退日志中捕获到异常,那么按照异常类型分类,最后按照关键堆栈里面的第一个非系统方法名进行分类。

参考文档

特别推荐–关于闪退收集框架的原理
https://www.cnblogs.com/smileEvday/p/Crash1.html
http://blog.csdn.net/a253143598/article/details/22289839
https://developer.apple.com/library/content/technotes/tn2151/_index.html

你可能感兴趣的:(iOS,Crash,iOS,崩溃日志)