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

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

iOS中的crash防护(一)unrecognized selector sent to instance的相关文章

CRASH: -[NSNull length]: unrecognized selector sent to instance错误的处理办法

开发中从后台请求数据,返回如下: 2014-12-05 16:44:52.535 掌麦[6984:613] getDefaultAddress: reuslt == { item =     { data =         { address = "<null>"; area = "<null>"; city = "<null>"; name = ""; phone = 185030513

iOS开发&mdash;&mdash;异常:[__NSCFNumber length]: unrecognized selector sent to instance

  *** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '-[__NSCFNumber length]: unrecognized selector sent to instance   网上参考的他人案例,与我的情况一模一样,所以直接盗用他的图和文章了. http://www.tuicool.com/articles/EZ3uMb7         这个错误是作者解析pilst文

iOS:编译错误[__NSDictionaryM objectAtIndexedSubscript:]: unrecognized selector sent to instance 0xa79e61

这个意思是,__NSDictionaryM  无法将值传到下标索引对象,言简意赅就是数组越界,但是再看看,这是数组吗?不是,所以,遇到这种crash,我这里有两种情况: 1.首先看看你 indexPath.row 用的有没有问题: 2.看看你请求下来的数据类型对不对. 如果还有其他情况,欢迎留言 有的时候,系统并不提示你crash在了哪里,仅仅返回到了main函数,这个时候,点击 show the Breakpoint Navigator,就是工程目录上面那一行工具中的断点符号,然后左下角有个"

iOS中常见 Crash 及解决方案

一.访问了一个已经被释放的对象 在不使用 ARC 的时候,内存要自己管理,这时重复或过早释放都有可能导致 Crash. 例子 NSObject * aObj = [[NSObject alloc] init]; [aObj release]; NSLog(@"%@", aObj); 原因 aObj 这个对象已经被释放,但是指针没有置空,这时访问这个指针指向的内存就会 Crash. 解决办法 使用前要判断非空,释放后要置空.正确的释放应该是: [aObj release]; aObj =

利用objc的runtime来定位次线程中unrecognized selector sent to instance的问题

昨天遇到一个只有一行错误信息的问题: -[NSNull objectForKey:]: unrecognized selector sent to instance 0x537e068 由于这个问题发生在次线程,所以没有太有用的堆栈信息,而是只有简单的SIGABRT信息: 考虑到unrecognized selector sent to instance这类问题是由于向某个对象发送了未实现的消息,这个过程大致如下(图片摘自这里): 参考Objective-C的对象模型: struct objc_

iOS 程序报错:reason: [NSArrayI addObject:]: unrecognized selector sent to instance

*** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '-[__NSArrayI addObject:]: unrecognized selector sent to instance 0x4b17be0' 错误:NSArray 不能使用 addObject:方法.可能是在程序运行的过程中,NSMutableArray转为了NSArray. 解决方案:核查数组,看看是否在NSArra

IOS 错误 [UIWebView cut:]: unrecognized selector sent to instance

那在什么场景中会出现这种情况呢? 如果一个包含文字的输入元素有焦点,然后按钮的点击会导致输入失去焦点,然后接下来在输入时双按会重新得到焦点并从弹出bar中选择剪切复制粘贴,就会导致此error. 也就是说当WebView页面中的HTML中有如下代码的时候 <input type="text"> <input type="button" > 即有输入框和按钮的时候,会出现 [UIWebView cut:]: unrecognized selec

静态库调用中“unrecognized selector sent to instance”错误

在开发调用静态库的中,出现 “unrecognized selector sent to instance 0x2b5f90”的错误 -[__NSCFConstantString xmlChar]: unrecognized selector sent to instance 0x2b5f90 *** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '-[__NSCFConstantS

ios [__NSCFNumber isEqualToString:]: unrecognized selector sent to instance 0x7a97d4c0&#39;报错

今天接口由get换成post,我去改进行登录但出现了这个错误,首先出错先看能不能与服务器交互,能不能获得数据,其次,获得的数据是不是你想要的,记住,首先出错要想到是自己的问题,还有就是程序崩了要学会自己解决,打断点找原因,一步步跟,看是哪里出了问题 我这个问题一看就是拿了NSCFNumber类型和NSString类型做了比较,由于不会响应isEqualToString方法而报错,所以只要转换一下就可以了 NSString *factory_id = [NSString stringWithFor