首页 > 代码库 > iOS中常见的 Crash 场景以及解决方法
iOS中常见的 Crash 场景以及解决方法
1. 常见的 Crash 场景
-
访问了僵尸对象
-
访问了不存在的方法
-
数组越界
-
在定时器下一次回调前将定时器释放,会Crash
2. 关于BAD_ACCESS
出现的原因: 访问了野指针, 比如访问已经释放对象的成员变量或者发消息, 死循环等;
解决方法:
1. 重写对象的respondsToSelector 方法, 先找到出现 EXECBADACCESS 前访问的最后一个 object;
2. 设置Enable Zombie Objects;
3. 设置全局断点快速定位问题代码所在行,接收所有异常;
4. Xcode7 之后已经集成了 BAD_ACCESS 捕获功能: Address Sanitizer 与步骤 2 一样设置;
5. analyze(静态分析, 不一定管用)
3. 什么时候会报 unrecognized selector 异常
-
当调用对象(子类,各级父类)中不含有对应方法的时候,并且依旧没有给出“消息转发”的具体方案的时候,程序在运行时会crash并抛出 unrecognized selector 异常
-
objective-c 中的每个方法在运行时会被转为消息发送objc_msgSend(reciver, selector)
-
例如 [person say]就会被转化为 objc_msgSend(person, @selector(say))
-
运行时会根据对象(reciever) 的isa 指针找到该对象所对应的类,然后会依次在对应的 类,父类,爷爷类,根类中找对应的方法
下面讲述对象方法的解析过程:
-
第一步:+(BOOL)resolveInstanceMethod:(SEL)sel实现方法,指定是否动态添加方法。 若返回NO,则进入下一步,若返回YES,则通过class_addMethod函数动态地添加方 法,消息得到处理,此流程完毕。
-
第二步:在第一步返回的是NO时,就会进入- (id)forwardingTargetForSelector:(SEL)aSelector方法,这是运行时给我们的第二次机会,用于指定哪个对象响应这个selector。不能指定为self。若返回nil,表示没有响应 者,则会进入第三步。若返回某个对象,则会调用该对象的方法。
-
第三步:若第二步返回的是nil,则我们首先要通过- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector指定方法签名,若返回nil,则表示不处 理。若返回方法签名,则会进入下一步。
-
第四步:当第三步返回方法方法签名后,就会调用- (void)forwardInvocation:(NSInvocation *)anInvocation方法,我们可以通过anInvocation 对象做很多处理,比如修改实现方法,修改响应对象等
-
第五步:若没有实现- (void)forwardInvocation:(NSInvocation *)anInvocation方法,那么 会进入- (void)doesNotRecognizeSelector:(SEL)aSelector方法。若我们没有实现这个方 法,那么就会crash,然后提示打不到响应的方法。到此,动态解析的流程就结束了。
4. 如何解决很难复现的crash
1. 有错误日志先看错误日志信息,
2.没有错误日志,第一步分析函数中的所有分支, 是否在语法上存在可能缺少条件的问题.所以检查所以的分支,确保每个分支执行的结果是正确的.
3.检查函数的参数,保证必传参数不能为空,若为空应该抛出异常,因此用断言检查参数的正确性很重要
4.检查函数中每个分支所调用的函数返回结果是正确的,其实就是递归过程(重复2,3步骤)
iOS中常见的 Crash 场景以及解决方法