专栏开篇: 在开发的过程中,作开为发者我们经常会遇到崩溃,闪退的情况,而且崩溃,闪退的情况有很多种。如果是在开发测试过程中的话,我们可以及时进行分析修复,但是对于我们的KPI还是会一有定的影响的,给导领留下的印象不佳。而且定位crash仍然需要花费很多的时间。如果崩溃,闪退发生在线上,那么对我们公司的产品影响更大,对我们的影响也是大的不行,轻则挨骂,重则扣工资。而且线上crash难以追踪定位,相信大家都深有体会。如果有一种机制,能够将常见的大多数crash给屏掉蔽,不会crash,而且可以发送crash信息到log日志或者服务器后台方便我们进行分析。不如仅此这种机制的核心代码在app运的行时候并不执行,只有在出现crash的情况才会执行,对性能的影响几乎未零。这么美妙的机制存在么?当然存在。接下来在这个专栏里,我将和大家一块探索,争取将这个机制完善起来。并心将核代码集成到JKCrashProtect
中,欢迎大家多多参与讨论哦。
unrecognized selector sent to instance
由于造崩成溃的原因很多,这一篇就只分析unrecognized selector sent to instance
这一种崩溃。这种崩种溃主要是找不到实现的方法时产生的,在开发的过程中产生的概率还是比较大的,尤其在团队规模较大,人员水平不一,有动态API的情下况比较容易产生。
对runtime不了解的小伙伴可以脑一补下runtime知识点。要想屏蔽掉这种crash,首先要知道方法是如何传递的。我这里做了简单的梳理,以demo 中点JKVC0
这个类为例,调用的方法是JKClicked
1,调用JKClicked
[JKVC0 JKClicked]; //如果找不到JKClicked 方法则执行后续的操作
2,调用父类的中的JKClicked方法
//如果父类中没有JKClicked方法,继续父类的父类调用JKClicked方法,直到NSObject调用JKClicked方法,如果NSObject也调用不到JKClicked方法,则执行下面的操作
3,动态方法解析
+ (BOOL)resolveInstanceMethod:(SEL)aSEL;//动态添加方法,但是如果我们没有动态添加JKClicked方法的实现的话,则行执下面的重定向操作。
4,消息转发
- (id)forwardingTargetForSelector:(SEL)aSelector//如果没进行消息转发操作,或者消息转发后仍然没有正常实现方法的调用,则行执后的面操作
5,崩溃
- (void)doesNotRecognizeSelector:(SEL)aSelector//调这用个方法同时会出现carsh,程序闪退。
如大果家对这个流程还不是太熟悉,我从网上找了来一个张图,大家可以看看。
知道了函数的调用流程,那我这边在消息转发的时候进行操作,在NSObject
的category
中对以下几个方法进行重写,具如体下:
-(id)forwardingTargetForSelector:(SEL)aSelector{
//将重向定后的消息接者收置为nil
return nil;
}
- (void)forwardInvocation:(NSInvocation *)anInvocation{
//将此方法进行重写,在里这不进行任何操作,屏蔽会产生crash的方法调用
}
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
NSString *methodName =NSStringFromSelector(aSelector);
if ([methodName hasPrefix:@"_"]) {//对私有方法不进行crash日志采集操作
return nil;
}
NSString *crashMessages = [NSString stringWithFormat:@"JKCrashProtect: [%@ %@]: unrecognized selector sent to instance",self,NSStringFromSelector(aSelector)];
NSMethodSignature *signature = [JKCrashProtect instanceMethodSignatureForSelector:@selector(JKCrashProtectCollectCrashMessages:)];
[[JKCrashProtect new] JKCrashProtectCollectCrashMessages:crashMessages];
return signature;//对methodSignatureForSelector 进行重写,不然不会调用forwardInvocation方法
}
其中JKCrashProtectCollectCrashMessages实如现下
:
- (void)JKCrashProtectCollectCrashMessages:(NSString *)crashMessage{
NSLog(@"%@",crashMessage);
}
核心代码展示完毕,接下来给大家演示下我的demo。
大家可以看到,tap手势和按钮,出发了不在存的方法,但是,并有没产生carsh,而是打印出了相关的log信息方便我们定位crash。
demo地址
cocoaPod:
pod "JKCrashProtect"