欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

iOS中的crash防护(一)unrecognized selector sent to instance

程序员文章站 2022-04-09 10:48:59
...

专栏开篇: 在开发的过程中,作开为发者我们经常会遇到崩溃,闪退的情况,而且崩溃,闪退的情况有很多种。如果是在开发测试过程中的话,我们可以及时进行分析修复,但是对于我们的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,程序闪退。

如大果家对这个流程还不是太熟悉,我从网上找了来一个张图,大家可以看看。iOS中的crash防护(一)unrecognized selector sent to instance
  知道了函数的调用流程,那我这边在消息转发的时候进行操作,在NSObjectcategory中对以下几个方法进行重写,具如体下:

-(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。
iOS中的crash防护(一)unrecognized selector sent to instance
大家可以看到,tap手势和按钮,出发了不在存的方法,但是,并有没产生carsh,而是打印出了相关的log信息方便我们定位crash。
demo地址
cocoaPod:

pod "JKCrashProtect"

更多优质文章,可以微信扫码关注:
iOS中的crash防护(一)unrecognized selector sent to instance