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"

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

  • 2
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在iOS开发,当我们运行程序时,有时候会遇到"appDelegate window: unrecognized selector sent to instance"的错误。这个错误发生的原因是我们使用了一个未定义的方法。 通常,这个错误是由于我们在我们的代码使用了一个不存在的方法。具体来说,"appDelegate"是我们的应用程序的代理类,"window"是一个窗口对象。这个错误的意思是我们在"appDelegate"实例上调用了一个名为"window"的方法,但是这个方法并不存在。 为了解决这个问题,我们需要检查我们的代码,找到在"appDelegate"实例上调用"window"方法的地方。一旦我们找到了这个地方,我们可以考虑以下几种解决方案: 1.确保我们正确地实例化了"window"对象。我们需要检查我们的代码,看看我们是否正确地创建了"window"对象并将其设置为"appDelegate"的属性。 2.检查我们的代码,确保我们没有在"appDelegate"类手动添加了一个名为"window"的方法。有时候我们可能会错误地将一个成员变量声明为一个方法,导致这个错误的发生。 3.如果我们通过Storyboard或XIB文件创建了窗口对象,我们需要确保我们正确地将窗口对象与"appDelegate"关联起来。可以通过检查我们的Storyboard或XIB文件的连接和引用关系来解决这个问题。 总结一下,当我们遇到"appDelegate window: unrecognized selector sent to instance"错误时,需要检查我们的代码,确保我们正确地实例化了窗口对象,并且没有使用一个未定义的方法。 ### 回答2: 这个错误通常是由于在代码调用了`[appDelegate window]`方法,但是`appDelegate`对象并不存在该方法所导致的。该方法的作用是返回`AppDelegate`对象的窗口属性。 产生此错误的原因可能有: 1. 在调用`[appDelegate window]`方法之前,没有正确初始化和分配内存给`appDelegate`对象。 2. `appDelegate`类没有定义`window`属性或对应的`getter`方法。 解决此问题的方法是: 1. 确保在使用`[appDelegate window]`方法之前正确初始化和分配内存给`appDelegate`对象,可以使用`alloc init`等方法。 2. 确保`appDelegate`类定义了`window`属性并有对应的`getter`方法。 以下是一个示例代码,演示了正确初始化`appDelegate`对象并调用`window`属性: ```objective-c // 创建并初始化AppDelegate对象 AppDelegate *appDelegate = [[AppDelegate alloc] init]; // 使用appDelegate对象的window属性 UIWindow *window = [appDelegate window]; ``` 希望以上回答能够解决你的问题。如果有任何进一步的问题,请随时提问。 ### 回答3: 这个错误通常是因为在使用iOS开发的AppDelegate时调用了window方法,但是实际上AppDelegate类并没有该方法,导致了这个错误。 解决这个问题有几种方法: 1. 检查调用window方法的地方是否正确。确保你正在调用的对象确实是AppDelegate的实例,并且确保没有拼写错误。 2. 检查你的AppDelegate类是否正确实现了UIApplicationDelegate协议。确认你的AppDelegate类有正确的UIApplicationDelegate方法实现,包括window属性的设置。 3. 检查你的Storyboard或XIB文件是否正确设置了AppDelegate的窗口。确认你的Storyboard或XIB文件已经正确设置了AppDelegate的窗口,以便在应用程序启动时正确地加载窗口。 如果你仔细检查并尝试了以上方法,仍然无法解决这个错误,那么可能是由于其他原因引起的。这种情况下,你可以尝试删除并重新创建AppDelegate类,或者重新创建项目。如果问题仍然存在,那么可能是其他代码或框架的错误导致的,需要进一步调试定位
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值