使用dSYM分析App崩溃日志

前言

我们在开发App过程中,因为连接到控制台,所以遇到问题会很容易找到问题代码。但是对于线上的App出现Crash的时候,我们不可能通过这种方式,也不现实,所以我们只能通过收集Crash信息,来解决Bug。而这种收集Crash信息并且分析定位到具体代码的第三方SDK很多。但是今天我们来自己实现一下。

收集 Crash 信息

Apple提供了NSException类来帮助我们收集异常信息。

NSException is used to implement exception handling and contains information about an exception — Apple Documentation.

点击这里来查看官方文档具体内容。

我们的确可以通过NSException来收集信息,但是,我们怎么把这个信息保存下来,并且上传到我们后台服务器,收集起来呢。这就需要用到另一个函数:NSUncaughtExceptionHandler

Sets the top-level error-handling function where you can perform last-minute logging before the program terminates.http://www.90168.org/

意思就是我们可以在App异常退出的之前有一分钟的时间来处理异常信息,利用这段时间,我们可以把Crash信息写入本地,也可以上传到服务器,但是考虑到网络阻塞原因,我们可能在这一分钟不能操作完毕,所以我们把上传放到下一次App启动时执行。

具体的代码如下:

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26
- (void)lyCarshLog {     [self uploadExceptionLog];     NSSetUncaughtExceptionHandler(&catchExceptionLog); } - (void)uploadExceptionLog {      if (log != nil) {         // 在这里上传 Crash 信息,上传完毕后要记得清空。      } } void catchExceptionLog(NSException *exception) {     // 获取 Crash 信息     NSArray *symbols = [exception callStackSymbols];     NSString *reason = [exception reason];     NSString *name = [exception name];     NSDictionary *userInfo = [exception userInfo];     //...          /*另外,我们可能需要一些别的信息,比如说发生 Crash 的设备的系统版本,设备型号,App的版本号*/     struct utsname systemInfo; // 需要导入`sys/utsname.h`头文件。     uname(&systemInfo);     NSString *deviceString = [NSString stringWithCString:systemInfo.machine encoding:NSUTF8StringEncoding];     NSDictionary *appInfo = [[NSBundle mainBundle] infoDictionary];     NSString *appVersion = [appInfo objectForKey:@"CFBundleShortVersionString"];     NSString *result = [NSString stringWithFormat:@"CarshReason = %@ \n name = %@ \n userInfo = %@ \n log = %@ \n systemVersion = %f \n deviceInfo = %@ \n appVersion = %@ ",reason,name,userInfo,symbols,[UIDevice currentDevice].systemVersion.floatValue,deviceString,appVersion];     // 把 result 写入本地。 }

Crash信息至此已经收集完毕,等待下次App启动的时候,我们把本地的Crash信息上传到服务器就OK了。

处理 Crash 信息 - 符号化(Fully Symbolicated)

我们得到的信息可能如下(Partially Symbolicated):

Tips: 堆栈跟踪是自下而上展示的,也就是最先调用的方法在最下面。

每行信息中包含的信息:

其中:

  1. Binary name 表明代码所在App或者Framework的位置。比如:line 0 是在CoreFoundation中,line 3 在CrashDemo中…
  2. Address 方法的内存地址。
  3. Class name 当前的类名。
  4. Method name 当前调用的方法名。
  5. Offset 相对加载地址/基地址(load address)的偏移量。

我们得到这个半符号化(Partially Symbolicated)的日志对我们分析Crash原因的帮助很有限,因为我们可能只能知道__NSArrayI objectAtIndex:调用出现了问题,但是不能定位到具体代码。所以我们要把它完全符号化(Fully Symbolicated)。

dSYM

我们需要借助dSYM来帮助我们完成符号化,对于dSYM文件的获取,我们可以通过多种方法,我这里只说一种:

先打开XcodeWindows->Organize->找到对应的app包,然后右键->Show in finder,找到appName. xcarchive->显示包内容->把dSYMs拷贝出来(或者就在里面操作)

atos

The atos command converts numeric addresses to their symbolic equivalents

我们使用atos命令来完成符号化,具体命令如下: $ atos -arch <Binary Architecture> -o <Path to dSYM file>/Contents/Resources/DWARF/<binary image name> -l <load address> <address to symbolicate> 其中:

  1. Binary Architecture: arm64armv6armv7 armv7s 根据自己的情况来写。
  2. Path to dSYM file: dSYM文件的路径。
  3. binary image name: 你工程的名字。
  4. load address: 基地址,如果我们的崩溃日志中没有这个信息(比如上面的Crash信息中就没有包含),就需要我们手动去计算这个load address:laod address = address to symbolicate - offset,比如:0x0000000102838119转化为十进制为4337139993,再减去偏移量265,为4337139728,在转化为十六进制0x0000000102838010
  5. address to symbolicate:当前方法的内存地址。

所以,上图为例:

1 2
$ cd CrashDemo/dSYMs $ atos -arch arm64 -o CrashDemo.app.dSYM/Contents/Resources/DWARF/CrashDemo -l  0x0000000102838010 0x0000000102838119

这时命令就会输出已经符号化过的信息: -[ViewController viewDidLoad] (in CrashDemo) (ViewController.m:45)

其中45就是问题代码在ViewController.m中的具体位置。

其实符号化的过程有多种方式,你可以参考Apple文档,对于其中UUID,只是为了我们找到App对应版本的dSYM文件,所以如果你能确定两者的对应,不需要我们再去获取。而且,使用上面方法,我们可以找到每一个版本对应的dSYM文件(假如你没有删除的话)。

最后

愉快的改Bug吧??

时间: 2025-01-13 18:53:49

使用dSYM分析App崩溃日志的相关文章

Xcode7.3工具解析App崩溃日志(.crash文件)

Xcode7.3工具解析App崩溃日志(.crash文件) 原文链接:http://blog.csdn.net/u011056605 开发的App或者游戏提交审核后,偶尔会收到测试反馈的消息,说应用崩溃了,bug偶尔出现,难以找到确定的重现方法. 怎么办?可以分析崩溃文件啊,也就是app崩溃后,自动保存在设备本地的.crash文件. 获得崩溃日志的方式,在 获取设备上的调试信息与崩溃日志分析 中有说. 在环境ok的情况下,xcode中是可以自动解析.crash文件的.旧版本的xcode甚至可以导

iOS系统app崩溃日志手动符号化

iOS系统app崩溃日志手动符号化步骤: 1.在桌面建立一个crash文件夹,将symbolicatecrash工具..crash文件..dSYM文件放到该文件夹中 a.如何查询symbolicatecrash路径位置? 执行命令:find /Applications/Xcode.app -name symbolicatecrash -type f 然后将symbolicatecrash复制一份 例如:Xcode7.3的symbolicatecrash路径 /Applications/Xcode

android app崩溃日志收集以及上传

源码获取请到github:https://github.com/DrJia/AndroidLogCollector 已经做成sdk的形式,源码已公开,源码看不懂的请自行google. 如果想定制适应自己app的sdk请自行fork. AndroidLogCollector android app崩溃日志收集sdk 1.0 作者:贾博士 崩溃日志收集方法: 1.LogCollector是lib包,在需要添加崩溃日志sdk的工程中导入此包. 2.导入lib后,在自己的工程的AndroidManife

如何通过友盟分析发布后App崩溃日志-b

要分析崩溃日志,首先需要保留发布时的编译出来的.xcarchive文件.这个文件包含了.DSYM文件. 我一般的做法是,发布成功后,把这个文件.xcarchive直接提交到代码版本库对应的版本分支里,这样就不会搞丢了. 这个文件在哪呢?打开XCode->菜单Window->Organizer,在编译成功的文件上右键,就能打开了. 两种比较麻烦的方法. 第一种方法: 使用dwarfdump命令 dwarfdump --uuid xx.app.dSYM     用来得到app的UUID.dwarf

iOS app 崩溃日志符号化及代码定位

1.什么是符号表? 符号表就是指在Xcode项目编译后,在编译生成的二进制文件.app的同级目录下生成的同名的.dSYM文件..dSYM文件其实是一个目录,在子目录中包含了一个16进制的保存函数地址映射信息的中转文件,所有Debug的symbols都在这个文件中(包括文件名.函数名.行号等),所以也称之为调试符号信息文件. 一般地,Xcode项目每次编译后,都会生成一个新的.dSYM文件.因此,App的每一个发布版本,都需要备份一个对应的.dSYM文件,以便后续调试定位问题. 注意:项目每一次编

iOS Crash 分析 符号化崩溃日志

参考: http://blog.csdn.net/diyagoanyhacker/article/details/41247367 http://blog.csdn.net/diyagoanyhacker/article/details/41247389 http://blog.csdn.net/diyagoanyhacker/article/details/41247411 http://www.cnblogs.com/smileEvday/p/Crash1.html 未符号化的崩溃日志就象一

使用atos分析iOS崩溃日志

当APP出现崩溃时,一般会在XCode的Device界面里面出现崩溃日志,大概像这样: Incident Identifier: FBF87174-405D-4F1C-A745-E3D9F7858F4F CrashReporter Key: 31da19c0a9879bc6ca613ad7611c14ccebe2a82d Hardware Model: iPhone4,1 Process: MyProj [285] Path: /var/mobile/Applications/C6E9AF9C-

app崩溃日志分析

find /Applications/Xcode.app -name symbolicatecrash -type f 拷贝symbolicatecrash 一般情况下,第一次使用symbolicatecrash会产生一个error,如下的错误信息 Error: "DEVELOPER_DIR" is not defined at /usr/local/bin/symbolicatecrash line 53. 解决办法是输入: export DEVELOPER_DIR="/A

iOS崩溃日志分析-b

1名词解释 1.1. UUID 一个字符串,在iOS上每个可执行文件或库文件都包含至少一个UUID,目的是为了唯一识别这个文件. 1.2. dwarfdump 苹果提供的命令行工具,其中一些功能就是查看可执行文件或库文件的UUID.示例: dwarfdump --uuid 应用名称.app/应用名称 dwarfdump --uuid 应用名称.dSYM 1.3. symbolicatecrash 苹果提供的命令行工具,可以将crash日志符号化为可读的堆栈信息.XCode6/XCode7版本中,