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

108 篇文章 1 订阅

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

如大果家对这个流程还不是太熟悉,我从网上找了来一个张图,大家可以看看。这里写图片描述
  知道了函数的调用流程,那我这边在消息转发的时候进行操作,在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。
这里写图片描述
大家可以看到,tap手势和按钮,出发了不在存的方法,但是,并有没产生carsh,而是打印出了相关的log信息方便我们定位crash。
demo地址
cocoaPod:

pod "JKCrashProtect"

更多优质文章,可以微信扫码关注:
这里写图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值