**
前言
**
移动APP中关于crash几乎是0容忍的,那么iOS中会有很多引起crash,比如最常见的数组越界,添加空值。
如果你想解决大部分可能引起的crash,保持代码的健壮性,又不想修改太多代码,那么推荐你使用WOCrashProtector
一款提高iOS稳定性,有效防止闪退的框架
WOCrashProtector
Release: 不会crash Debug: 为了更好的追踪问题,依旧会crash,但提供了更多的crash信息
常见carsh汇总
接下来介绍一下,哪些方法使用不慎会引起crash,当然就像上面说的,引入YCBStability会让你的代码更健壮,我在runtime中,进行了方法拦截,避免了carsh
因此,你只要导入YCBStability,不需要添加任何代码,你的代码将变得足够强壮
原理介绍
前面写了很多关于YCBStability解决crash问题,如果你只是想解决代码中的问题,读到这里就够了。
但如果你想了解更多知识,下面我来对crash进行汇总,希望能帮助到你。
OC方法中的那些坑
OB提供的方法并不是健壮的,最常见的,set一个nil or数组越界都会引起crash,有很多朋友喜欢加一些判断,我认为这是一个良好的意识,但并不是最好的做法。
考虑到代码的简洁性,我建议你删掉这些if-else,引入YCBStability
但如果你已经用了Category来避免这些方法引起的crash,那么我建议你继续Category
NSArray
方法 crash说明 :
- (ObjectType)objectAtIndex:(NSUInteger)index 当index大于数组count的时候引起数组越界
- (NSUInteger)indexOfObject:(ObjectType)anObject 当anObject为nil时carsh
NSMutableArray
方法 crash说明
- (void)addObject:(ObjectType)anObject 当anObject为nil时carsh
- (void)insertObject:(ObjectType)anObject atIndex:(NSUInteger)index 当anObject为nil时carsh,当index>mutArray.count时,会产生越界
- (void)removeObjectAtIndex:(NSUInteger)index 当index>mutArray.count时,会产生越界
NSMutableSet
方法 crash说明
- (void)addObject:(ObjectType)object 当anObject为nil时carsh
NSMutableDictionary
方法 crash说明
- (void)setObject:(ObjectType)anObject forKey:(KeyType)aKey 当anObject为nil时,或key为nil时,都会引起crash
泛型的坑
在NSDictionary中,我们经常用到这个方法
- (nullable ObjectType)objectForKey:(KeyType)aKey;
这种返回值类型很容易产生坑,举个列子:
项目API文档服务器返回数据如下
{
list :(
‘a’,
‘b’
)
}
这时候,你通过
NSArray *array = [dic objectForKey:@”list”];
NSString *str = [array firstObject];
正常状态下,不会有任何问题,但是你要知道,服务器的数据是不可信的,有一天服务器代码出现了bug,list不再是数组,那么APP就会Crash
{
list: {}
}
因为此时的[dic objectForKey:@”list”];取出返回了Dic, 而你依然认为是NSArray,并且调用了firstObject方法,Dic里没有firstObject,APP会crash
面对服务器返回数据的信任问题,有些同学养成了好习惯,加入了if-else,但是我希望你引入YCBStability,使用如下方法:
@interface NSDictionary (YCBStability)
/* 取key对应的字符串 /
- (NSString *)getStringForKey:(id)key;
/* 取key对应的数组 /
- (NSArray *)getArrayForKey:(id)key;
/* 取key对应的字典 /
- (NSDictionary *)getDictinaryForKey:(id)key;
(int)getIntForKey:(id)key;
(float)getFloatForKey:(id)key;
(BOOL)getBoolForKey:(id)key;
@end
根据你要的类型选择不同的方法,
debug模式:如果和预期类型不符会crash,方便问题追踪 release模式:我做了容错处理,避免了crash
非空判断
为了更好的支持类型判断,我提供了如下方法供使用
@interface YCBNonEmpty : NSObject
/* 判断是否是非空的数组 /
+ (BOOL)isArray:(id)object;
/* 判断是否是非空的集合/
+ (BOOL)isSet:(id)object;
/* 判断是否是非空的字符串 /
+ (BOOL)isString:(id)text;
/* 判断是否是非空的字典 /
+ (BOOL)isDictionary:(id)object;
@end
非空判断是代码中经常用到的技巧,以保证代码的健壮性,在YCBNonEmpty中,我依据类型和count两个条件,判断非空,更加安全准确。