但凡开发过ios程序的,都碰到过程序崩溃,而信息没有办法及时收集的问题。目前网上可以搜到的收集崩溃信息的方式一般是通过设置一个NSUncaughtExceptionHandler的回调函数进行捕获,然后在回调函数中把堆栈信息存到本地,再选择一个合适的时机上传到服务器。这种方式可以捕捉到大部分的异常,但存在一个盲区,就是像数组越界等这一类的内存错误的异常是无法捕捉的。如果是在平时开发中,这种问题会及时在XCdoe中抛出来,但应用上线后这种问题就会漏掉了。
为了解决这个问题,我研究了一下苹果的帮助文档,并做了一些实验,发现系统提供的try...catch函数可以捕捉到这类异常,例如:
但是,问题是我们不能在所有代码中都加上try...catch...,这又不是开发java程序对吧……
那么,有没有比较巧妙的方式解决这个问题呢?
其实问题的关键在于能不能捕捉到exception对象,因为他里面包含了你想要的所有信息。我想到了添加一个NSException的Category,看了看NSException的头文件,貌似只有raiseXXX这样的方法,那假如我们重写raise方法,能不能捕获到异常呢?带着这个疑问,我试了下,果然崩溃的时候走进了这个方法。
但是光重写raise方法是不够的,因为拦截了系统这个方法,不会引起崩溃了。
那怎么办呢,我又想到了AOP(通过方法置换的原理在原有方法执行之前或之后插入一段自己的代码),最终结果如下:
这里我用到了Aspect这个库进行方法置换,在exception 的raise方法执行前,我拿到exception,然后传给异常处理组件,进行保存和上传。这样,不管是什么类型的崩溃,都能轻松的获得他的崩溃信息。