线上APP的崩溃率一直是衡量APP用户体验的重要条件之一,所以,我们很有必要做一些安全防护,让APP尽可能少的产生Crash,提高用户体验。在以前的项目中零零散散做过一些防护,这次专门为平台封装了一个Pod库,供各个业务线直接引用,降低线上APP崩溃率,并将错误信息上传到服务器进行分析。

     其实,在开发过程中我们通过设置Xcode配置项、代码里边做判断、加宏编译等可以发现和避免很多Crash,如代码里边使用respondsToSelector、@available做系统判断等,这个库做的防护主要是代码在运行过程中动态产生的Crash,比如接口返回的数据类型有问题导致产生unrecognized selector、NSString和NSArray越界、KVC和NSDictionary访问key值和设置value值等问题。

     GitHub上也有一些封装好的三方库,但是,由于长时间不进行维护了,在新的系统和机型上可能会产生问题,同时,随着iOS开发语言和系统的更新,之前一些容易产生Crash的问题也修复了,所以没必要在进行防护了,比如对象dealloc之后NSNotification还存在会导致崩溃的问题,以及KVO的一些崩溃问题等。

     在这里我先列举一些常见的Crash问题,然后分类型进行分析,列举最新的防护手段(因为有的已经不需要进行防护了):

1、低系统使用高系统API产生的Crash。

2、unrecognized selector类型的Crash,尤其接口返回数据类型有匹配的情况下产生的。

3、NSString、NSMutableString及相关类簇产生的Crash。

4、NSArray、NSMutableArray及相关类簇产生的Crash。

5、NSDictionary、NSMutableDictionary及相关类簇产生的Crash。

6、使用KVC产生Crash。

7、KVO相关Crash。

8、NSNotification相关Crash。

9、NSTimer Crash。

10、野指针 Crash。

11、非主线程刷新UI 导致的Crash。

 

防护手段:

1、低系统使用高系统API产生的Crash:

这个在Xcode中进入工程的Build Settings页面,在“Other C Flags”和“Other C++ Flags”中增加“-Wunguarded-availablility”,设置好之后,如果误调用了高版本API,Clang会检测到并报出警告。

2、unrecognized selector类型的Crash:

方法找不到这种Crash,在崩溃日志中可以说占了很大的比重,尤其是接口返回的数据没有按约定的类型返回或者返回了一个NSNull对象极易产生崩溃。

解决办法:通过runtime来hook NSObject的方法,在自己的方法里边加入try catch去调用原始的方法,这里利用runtime消息转发三部曲机制来处理,这里思考一下,三个方法中我们应该选择哪个方法去做防护呢?第一个方法resolveInstanceMethod中动态增加方法不太合适,第二个方法forwardingTargetForSelector将消息转发给别的对象其实是可以的,但是如果在这个方法中实现,团队中有人想让消息转发走到第三步(forwardInvocation方法)在第三步中做自己的处理,那就会产生问题,因为根本不会走到第三步就被拦截了。所以,为了尽量减少对项目产生侵入性,我选择在第三步对forwardInvocation和methodSignatureForSelector方法做处理来截获将要产生的异常,虽然第三步开销大一些吧。

注意:这里不能对所有的NSObject类都进行这个防护,因为经过测试发现,系统的一些UIView类会在消息转发第三步做自己的处理,比如UIKeyboard相关的。目前,我主要是对@[@"NSNull",@"NSNumber",@"NSString",@"NSDictionary",@"NSArray",@"NSSet"] 这些OC中基础类做了处理。

那对于实现就很简单了,通过hook NSObject的forwardInvocation实例方法和methodSignatureForSelector实例方法,在hook的methodSignatureForSelector方法中构造一个自定义的方法签名,在hook的forwardInvocation方法中加上try catch防护来拦截异常,大概的代码如下:

- (NSMethodSignature *)avoidCrashMethodSignatureForSelector:(SEL)aSelector {
    NSMethodSignature *ms = [self avoidCrashMethodSignatureForSelector:aSelector];
    if (ms == nil) {
        //只对预设置的类构造NSMethodSignature
        for (NSString *classStr in classNameArray) {
            if ([self isKindOfClass:NSClassFromString(classStr)]) {
                //emptyMethod方法是空实现,主要是为了构造一个方法签名
                ms = [AvoidCrashTools instanceMethodSignatureForSelector:@selector(emptyMethod)];
                break;
            }
        }
    }
    return ms;
}

- (void)avoidCrashForwardInvocation:(NSInvocation *)anInvocation {
    @try {
        [self avoidCrashForwardInvocation:anInvocation];
    } @catch (NSException *exception) {
        //解析堆栈,上传错误信息
        [AvoidCrashTools noteErrorWithException:exception type:TypeUnrecognizedSelector];
    } @finally {
    }
}

 

3、NSString、NSMutableString相关方法的Crash:

这里需要注意一下,由于类簇的存在,在hook相关方法的时候,需要考虑类簇,将类簇的相关方法也要hook,比如__NSCFConstantString、 NSTaggedPointerString、 __NSCFString等。我目前防护的相关方法有:

1. - (unichar)characterAtIndex:(NSUInteger)index
2. - (NSString *)substringFromIndex:(NSUInteger)from
3. - (NSString *)substringToIndex:(NSUInteger)to {
4. - (NSString *)substringWithRange:(NSRange)range {
5. - (NSString *)stringByReplacingOccurrencesOfString:(NSString *)target withString:(NSString *)replacement
6. - (NSString *)stringByReplacingOccurrencesOfString:(NSString *)target withString:(NSString *)replacement options:(NSStringCompareOptions)options range:(NSRange)searchRange
7. - (NSString *)stringByReplacingCharactersInRange:(NSRange)range withString:(NSString *)replacement

NSMutableString防护的方法有:

由于NSMutableString是继承于NSString,所以这里和NSString有些同样的方法就不重复写了
1. - (void)replaceCharactersInRange:(NSRange)range withString:(NSString *)aString
2. - (void)insertString:(NSString *)aString atIndex:(NSUInteger)loc
3. - (void)deleteCharactersInRange:(NSRange)range

 

4、NSArray、NSMutableArray及相关类簇产生的Crash。

5、NSDictionary、NSMutableDictionary及相关类簇产生的Crash。

4和5 容器类的防护,没啥可说的,和3中NSString的防护是类似的,具体要hook的方法自己进行选择就可以了,注意进行自测。

 

6、使用KVC产生Crash:

使用KVC访问或设置不存在的key或者访问和设置nil等会产生Crash。所以需要进行防护,防护的方法有:

1、setValue:forKey:
2、setValue:forKeyPath:
3、setValue:forUndefinedKey:
4、setValuesForKeysWithDictionary:

5、valueForKey
6、valueForKeyPath
7、valueForUndefinedKey

7、KVO相关Crash:

现在当被观察者dealloc的时候还被监听着,并不会产生Crash。但是,重复移除观察者或者KVO注册观察者与移除观察者不匹配还是会产生Crash的,不过这两种情景在开发过程中很容易就被发现了,所以,没有必要再做防护了。

8、NSNotification相关Crash:

在iOS9之前当一个对象添加了notification之后,如果dealloc的时候,仍然持有notification,就会出现NSNotification类型的crash。苹果在iOS9之后专门针对于这种情况做了处理,所以在iOS9之后,即使开发者没有移除observer,Notification crash也不会再产生了。如果APP目前从iOS9开始适配的话,NSNotification的Crash也是可以忽略了。

9、NSTimer Crash:

在日常开发中大家会经常使用到NSTimer,但使用NSTimer的 scheduledTimerWithTimeInterval:target:selector:userInfo:repeats:接口做重复性的定时任务时存在一个问题:NSTimer会强引用target实例,所以需要在合适的时机invalidate定时器,否则就会由于定时器timer强引用target的关系导致target不能被释放,造成内存泄露,甚至在定时任务触发时导致crash。 crash的展现形式和具体的target执行的selector有关。

关于NSTimer的防Crash和防循环引用,可以用一个NSProxy对象来做桥接,弱引用原来的target,把这个NSProxy对象当做NSTimer的第一个参数传进去,当调用方法的时候还是让原来的target去调用。这样NSTimer强持有NSProxy对象,但是NSProxy对象是弱持有原来的target,所以就解除循环引用了。可以参考YYText中的YYTextWeakProxy类的设计。

10、野指针 Crash:

在目前的ARC时代,发生野指针的情况很少见了,所以没必要专门做野指针相关的防护了。我们可以设置Xcode的配置,在开发过程中自动检测是否有野指针的情况存在,即利用僵尸对象(Zombie Objects)检测工具来检测,设置如下:

iOS crash治理 ios crash防护_runtime

11、非主线程刷新UI 导致的Crash:

由于UIKit是非线程安全的,在子线程刷新UI的相关操作可能会引起Crash,这个我们也可以通过配置Xcode环境变量来在开发的过程中发现这类问题,配置如下,在Edit Scheme中勾选Main Thread Checker 即可:

iOS crash治理 ios crash防护_降低崩溃率_02

 

至此,上边列举的常见Crash都有了预防方案,这些方案需要根据自己的项目进行选择。

 

这里说明一下,当本来会产生的Crash被我们拦截之后,我们需要上传这些信息到我们的服务器,这样才能发现这些问题并及时的修复。所以,当在try catch中产生异常时,需要解析当前堆栈信息,找到崩溃的具体类和具体方法,上传这些异常信息。

解析堆栈信息我参考了别人的代码同时加入了自己的优化代码,大概如下:

+ (void)noteErrorWithException:(NSException *)exception type:(CrashType)type {
    //堆栈数据
    NSArray *callStackSymbolsArr = [NSThread callStackSymbols];
    
    //获取在哪个类的哪个方法中实例化的数组  字符串格式 -[类名 方法名]  或者 +[类名 方法名]
    NSString *mainCallStackSymbolMsg = [AvoidCrashTools getMainCallStackSymbolMessageWithCallStackSymbols:callStackSymbolsArr];
    if (mainCallStackSymbolMsg == nil) {
        //没有找到具体的崩溃地址,则默认上传前10条堆栈信息
        NSInteger max = callStackSymbolsArr.count >= 10 ? 10 : callStackSymbolsArr.count;
        NSMutableString *muStr = [NSMutableString string];
        for (int i = 0; i < max; i++) {
            [muStr appendString:callStackSymbolsArr[i]];
        }
        mainCallStackSymbolMsg = muStr;
    }
        
    NSMutableDictionary *infoDic = [NSMutableDictionary dictionary];
    [infoDic setValue:exception.name forKey:@"errorName"];
    [infoDic setValue:[AvoidCrashTools typeStrWithCrashType:type] forKey:@"crashType"];
    [infoDic setValue:exception.reason forKey:@"errorReason"];
    [infoDic setValue:mainCallStackSymbolMsg forKey:@"errorPlace"];

    AvoidCrashLog(@"Crash = %@",infoDic);
    
    //将错误信息放在字典里,用通知的形式发送出去
    dispatch_async(dispatch_get_main_queue(), ^{
        [[NSNotificationCenter defaultCenter] postNotificationName:AvoidCrashNotification object:infoDic userInfo:nil];
    });
}

/**
 *  获取Crash的具体类和方法<根据正则表达式匹配出来>
 *
 *  @param callStackSymbols 堆栈主要崩溃信息
 *
 *  @return Crash的地方
 */
+ (NSString *)getMainCallStackSymbolMessageWithCallStackSymbols:(NSArray<NSString *> *)callStackSymbols {
    //mainCallStackSymbolMsg的格式为   +[类名 方法名]  或者 -[类名 方法名]
    __block NSString *mainCallStackSymbolMsg = nil;
    
    //匹配出来的格式为 +[类名 方法名]  或者 -[类名 方法名]
    NSString *regularExpStr = @"[-\\+]\\[.+\\]";
    NSRegularExpression *regularExp = [[NSRegularExpression alloc] initWithPattern:regularExpStr options:NSRegularExpressionCaseInsensitive error:nil];
    
    for (int index = 0; index < callStackSymbols.count; index++) {
        NSString *callStackSymbol = callStackSymbols[index];
        [regularExp enumerateMatchesInString:callStackSymbol options:NSMatchingReportProgress range:NSMakeRange(0, callStackSymbol.length) usingBlock:^(NSTextCheckingResult * _Nullable result, NSMatchingFlags flags, BOOL * _Nonnull stop) {
            if (result) {
                NSString* tempCallStackSymbolMsg = [callStackSymbol substringWithRange:result.range];
                //get className
                NSString *className = [tempCallStackSymbolMsg componentsSeparatedByString:@" "].firstObject;
                className = [className componentsSeparatedByString:@"["].lastObject;
                NSBundle *bundle = [NSBundle bundleForClass:NSClassFromString(className)];
                //filter category and system class
                if (![className hasSuffix:@")"] && bundle == [NSBundle mainBundle]) {
                    mainCallStackSymbolMsg = tempCallStackSymbolMsg;
                }
                *stop = YES;
            }
        }];
        //除去AvoidCrash本身的类
        if (mainCallStackSymbolMsg.length && ![mainCallStackSymbolMsg containsString:@"AvoidCrash"]) {
            break;
        }
    }
    return mainCallStackSymbolMsg;
}

+ (NSString *)typeStrWithCrashType:(CrashType)type {
    NSString *typeStr;
    switch (type) {
        case TypeUnrecognizedSelector:
            typeStr = @"Crash_UnrecognizedSelector";
            break;
        case TypeKVC:
            typeStr = @"Crash_KVC";
            break;
        case TypeNSString:
            typeStr = @"Crash_NSString";
            break;
        case TypeNSMutableString:
            typeStr = @"Crash_NSMutableString";
            break;
        default:
            break;
    }
    return typeStr;
}

 

解析完堆栈,拼装这些异常信息,最终得到如下的崩溃信息,之后进行上传就可以了:

注意:如果这里errorPlace(崩溃的具体位置)解析不到,则默认会上传前10条堆栈信息,之后从服务器拿到这里的堆栈信息后,配合dsym文件自己进行解析就可以了。

info = {
    crashType = "Crash_UnrecognizedSelector";
    errorName = NSInvalidArgumentException;
    errorPlace = "-[ViewController viewDidLoad]";
    errorReason = "-[__NSCFNumber count]: unrecognized selector sent to instance 0xbd806617e7edf3ae";
}

 

至此,这个库基本就完成了,下面在项目中调用即可:

#ifndef DEBUG
    //开启防Crash处理
    [[AvoidCrash sharedInstance] openAllAvoidCrash];

    //info里边包含发生的崩溃的原因,崩溃的类型和崩溃的具体位置
    [AvoidCrash sharedInstance].reportBlock = ^(NSDictionary * _Nullable info) {
        dispatch_async(dispatch_get_global_queue(0, 0), ^{
            //在这里将拦截的异常上报服务器
            ...
        });
    };
#endif