AvoidCrash集成和简单封装 unrecognized selector sent to instance 崩溃拦截

关于后台返回数据类型与接口文档类型不一致的等一些问题

项目足够成熟的情况,一般很少出现crash问题。基本上大部分崩溃出现在对接后台数据时出现的数据类型不一致,以及数组越界、字典赋空值的问题。我之前也问过几个朋友,通过runtime更换系统方法拦截数组越界这样的问题确实很有效。

但是还会存在更麻烦的问题,与PHP数据对接时(Java好像没有这个问题哦,你们PHP不是最好的语言吗),数组或字典类型数据在没有值的情况下会返回一个bool值或者空值,如果是我们所熟知的nil值,一般在取值是不会出现崩溃,但是经过几次问题捕获到的错误日志,能发现AFNetworking解析数据时,由于NSDictionarykey值和value值都是不能为nil的,并且数组和字典都只能存放对象元素,所以这两种值会被转换成NSNull对象或者NSNumber对象,如果作为数组或字典对象来取值的话就会造成 unrecognized selector sent to instance崩溃。

大致上是这样一个场景
    //模拟网络解析到错误类型数据
    NSArray *array = @(true);
    NSDictionary *dic = @[@"qq",@"ww"];
    //取值
    NSString *string = array[2];
    NSString *st = dic [@"abc"];
    NSInteger c = array.count;
    NSLog(@"string==%@\nc == %ld\nst == %@",string,c,st);

PS因为遇见过的次数比较多,为了防止类型不同造成崩溃,基本上我的cell赋值都是这样的:

self.nameLab.text = [NSString stringWithFormat:@"%@",model.name ?: @""];

寻找了很多这类崩溃的解决办法,我找到了AvoidCrash 工具,还参考了这篇文章。集成方法确实很简单,我在这里自己封装了一个工具类,分担AppDelegate的业务逻辑。

调用方法
#import "FDGAvoidCrashManager.h"//捕获异常机制

...

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
  //捕获并拦截异常
    [FDGAvoidCrashManager sharedFDGACM];
  return YES;
}
封装代码

FDGAvoidCrashManager.h

#import 

@interface FDGAvoidCrashManager : NSObject
/**
 * 单例初始化 运行崩溃拦截 捕获异常报错
 */
+ (instancetype)sharedFDGACM;
@end

FDGAvoidCrashManager.m

#import "FDGAvoidCrashManager.h"
#import //腾讯bugl
#import "AvoidCrash.h"//捕获异常机制

#define Bugly_ErrorName_AvoidCrash @"AvoidCrash拦截的异常"
@implementation FDGAvoidCrashManager

static dispatch_once_t fdgACMOnce;
static FDGAvoidCrashManager *instance;
+ (instancetype)sharedFDGACM {
    dispatch_once(&fdgACMOnce, ^{
        instance = [[FDGAvoidCrashManager alloc] init];
        [instance registAvoidCrash];
    });
    return instance;
}
#pragma mark 防崩机制AvoidCrash
- (void)registAvoidCrash {
    //启动防止崩溃功能(注意区分becomeEffective和makeAllEffective的区别)
    //具体区别请看 AvoidCrash.h中的描述
    //建议在didFinishLaunchingWithOptions最初始位置调用 上面的方法

    [AvoidCrash makeAllEffective];

    //================================================
    //   1、unrecognized selector sent to instance(方式1)
    //================================================


    //若出现unrecognized selector sent to instance导致的崩溃并且控制台输出:
    //-[__NSCFConstantString initWithName:age:height:weight:]: unrecognized selector sent to instance
    //你可以将@"__NSCFConstantString"添加到如下数组中,当然,你也可以将它的父类添加到下面数组中
    //比如,对于部分字符串,继承关系如下
    //__NSCFConstantString --> __NSCFString --> NSMutableString --> NSString
    //你可以将上面四个类随意一个添加到下面的数组中,建议直接填入 NSString

    //我所开发的项目中所防止unrecognized selector sent to instance的类有下面几个,主要是防止后台数据格式错乱导致的崩溃。个人觉得若要防止后台接口数据错乱,用下面的几个类即可。
    //NSStringFromClass(NSString.class);
    NSArray *noneSelClassStrings = @[
                                     @"NSNull",
                                     @"NSNumber",
                                     @"NSString",
                                     @"NSDictionary",
                                     @"NSArray"
                                     ];
    [AvoidCrash setupNoneSelClassStringsArr:noneSelClassStrings];


    //================================================
    //   2、unrecognized selector sent to instance(方式2)
    //================================================

    //若需要防止某个前缀的类的unrecognized selector sent to instance
    //比如你所开发项目中使用的类的前缀:CC、DD
    //你可以调用如下方法
    //NSArray *noneSelClassPrefix = @[
    //                                @"CC",
    //                                @"DD"
    //                                ];
    //[AvoidCrash setupNoneSelClassStringPrefixsArr:noneSelClassPrefix];

    //监听通知:AvoidCrashNotification, 获取AvoidCrash捕获的崩溃日志的详细信息
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(dealwithCrashMessage:) name:AvoidCrashNotification object:nil];
}

/**
 * 监听捕获到的异常
 */
- (void)dealwithCrashMessage:(NSNotification *)note {
    //注意:所有的信息都在userInfo中
    //你可以在这里收集相应的崩溃信息进行相应的处理(比如传到自己服务器)

    //异常拦截并且通过bugly上报
    NSDictionary *info = note.userInfo;
    NSString *errorReason = [NSString stringWithFormat:@"异常拦截\n【Error原因】%@========\n【Error位置】%@========\n【解决方式】%@========\n【错误名】%@", info[@"errorReason"], info[@"errorPlace"], info[@"defaultToDo"], info[@"errorName"]];
    NSArray *callStack = info[@"callStackSymbols"];

    //存储拦截日志
    [YQHttpTool logWithStat:@"崩溃拦截" host:@"FDGAC异常拦截" parmeats:errorReason result:info];

    ///如果集成了bugly 将不会上报异常 需要手动上传错误
    [FDGAvoidCrashManager reportErrorName:Bugly_ErrorName_AvoidCrash errorReason:errorReason callStack:callStack extraInfo:nil];

}
/** bugly 上报错误信息 */
+ (void)reportErrorName:(NSString *)errorName errorReason:(NSString *)errorReason callStack:(NSArray *)aStackArray extraInfo:(NSDictionary *)info{

    [Bugly reportExceptionWithCategory:3 name:errorName reason:errorReason callStack:aStackArray extraInfo:info terminateApp:NO];
}

@end

你可能感兴趣的:(Crash,崩溃拦截,iOS,unrecognized,selector,sent,to,数组越界,Crash,崩溃拦截)