iOS 降低 NSArray Crash 风险

在日常开发中,会存在以下应用场景

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        NSString *string = nil;
        NSArray *array = @[string];
        
        NSLog(@"%@", array);
    }
    return 0;
}

毫无疑问,这样的代码跑起来,会直接crash,那么我们有没有办法通过代码的形式,让这种场景进行可执行又不crash呢?答案是有的,可以通过runtime的方式,将我们自己的方法跟系统方法进行互换,从而达到我们所要的效果。那么问题又来了,怎么知道@[]这个是调用的哪个方法呢?我们可以用Clang将OC代码转成C++代码。
Clang之后代码如下

int main(int argc, const char * argv[]) {
    /* @autoreleasepool */ { __AtAutoreleasePool __autoreleasepool; 
        NSString *string = __null;
        NSArray *array = ((NSArray *(*)(Class, SEL, ObjectType  _Nonnull const * _Nonnull, NSUInteger))(void *)objc_msgSend)(objc_getClass("NSArray"), sel_registerName("arrayWithObjects:count:"), (const id *)__NSContainer_literal(1U, string).arr, 1U);

        NSLog((NSString *)&__NSConstantStringImpl__var_folders_8f_y3mvcz2d4_s228xbxgg4ky0h0000gp_T_main_1c0812_mi_0, array);
    }
    return 0;
}

经过clang,可以发现@[]这种方式创建的数组是通过发送消息给NSArray执行arrayWithObjects:count:这个方法来创建的数组。

方法拿到了,下面就通过创建分类+运行时的方式,将系统方法和自定义方法进行互换。

新建一个NSArrayCategory文件

导入 #import <objc/runtime.h>

在实现文件中,重写 load 方法,将系统方法和自定义方法进行替换

+ (void)load {
    
    Method system_arrayWithObjectsCountMethod = class_getClassMethod(self, @selector(arrayWithObjects:count:));
    Method my_arrayWithObjectsCountMethod = class_getClassMethod(self, @selector(yxc_arrayWithObjects:count:));
    method_exchangeImplementations(system_arrayWithObjectsCountMethod, my_arrayWithObjectsCountMethod);
}

自定义 yxc_arrayWithObjects:count: 方法

/**
 @[] 字面量初始化调用方法

 @param objects 对象
 @param cnt 数组个数
 @return 数组
 */
+ (instancetype)yxc_arrayWithObjects:(id  _Nonnull const [])objects count:(NSUInteger)cnt {
    
    NSMutableArray *objectArray = [NSMutableArray array];
    
    for (int i = 0; i < cnt; i++) {
        id object = objects[i];
        if (object && ![object isKindOfClass:[NSNull class]]) {
            [objectArray addObject:object];
        }
    }
    
    return [NSArray arrayWithArray:objectArray];
}

编译代码,发现这时候main函数中同样的代码不会再崩溃了,这样我们通过 Category + runtime 的方式达到了降低crash的风险。

接下来,我们来看另外一种情况

main函数代码改成如下:

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        NSArray *array = @[@1, @2, @3];
        NSLog(@"%@", array[7]);
    }
    return 0;
}

array数组只有三个元素,这边输出第八个元素,数组已经越界了,实际上在开发中,经常出现数组越界访问的情况,那么我们又该怎么去降低crash的风险呢?
如果按照刚才的形式,先进行 clang ,然后再通过 category + runtime进行转换

clang

int main(int argc, const char * argv[]) {
    /* @autoreleasepool */ { __AtAutoreleasePool __autoreleasepool; 
        NSArray *array = ((NSArray *(*)(Class, SEL, ObjectType  _Nonnull const * _Nonnull, NSUInteger))(void *)objc_msgSend)(objc_getClass("NSArray"), sel_registerName("arrayWithObjects:count:"), (const id *)__NSContainer_literal(3U, ((NSNumber *(*)(Class, SEL, int))(void *)objc_msgSend)(objc_getClass("NSNumber"), sel_registerName("numberWithInt:"), 1), ((NSNumber *(*)(Class, SEL, int))(void *)objc_msgSend)(objc_getClass("NSNumber"), sel_registerName("numberWithInt:"), 2), ((NSNumber *(*)(Class, SEL, int))(void *)objc_msgSend)(objc_getClass("NSNumber"), sel_registerName("numberWithInt:"), 3)).arr, 3U);
        NSLog((NSString *)&__NSConstantStringImpl__var_folders_8f_y3mvcz2d4_s228xbxgg4ky0h0000gp_T_main_e7346f_mi_0, ((id (*)(id, SEL, NSUInteger))(void *)objc_msgSend)((id)array, sel_registerName("objectAtIndexedSubscript:"), (NSUInteger)7));
    }
    return 0;
}

替换 objectAtIndexedSubscript: 系统方法

Method system_objectAtIndexedSubscriptMethod = class_getInstanceMethod(self, @selector(objectAtIndexedSubscript:));
Method my_objectAtIndexedSubscriptMethod = class_getInstanceMethod(self, @selector(yxc_objectAtIndexedSubscript:));
method_exchangeImplementations(system_objectAtIndexedSubscriptMethod, my_objectAtIndexedSubscriptMethod);

yxc_objectAtIndexedSubscript方法

/**
 @[] 形式获取数组对象

 @param idx 数组下标
 */
- (id)yxc_objectAtIndexedSubscript:(NSUInteger)idx {
    
    if (idx >= self.count) return nil;
    
    return [self objectAtIndex:idx];
}

然后编译运行,发现还是崩溃,同样的方式,为什么不同的结果?
这时候我们需要跳入Fundation查看 yxc_objectAtIndexedSubscript 这个方法,发现这个方法并不是 NSArray 原有的方法,是 NSArray一个名为NSExtendedArray的分类方法,所以才导致我们这种方式无效。这时候,我们可以查看崩溃信息.

*** Terminating app due to uncaught exception 'NSRangeException', reason: '*** -[__NSArrayI objectAtIndexedSubscript:]: index 7 beyond bounds [0 .. 2]'

这时候发现,系统提示的是 __NSArrayIobjectAtIndexedSubscript: 方法崩溃。此时提示的是 __NSArrayI 而不是 NSArray,所以这时候我们把load方法替换改成

Method systemMethod1 = class_getInstanceMethod(NSClassFromString(@"__NSArrayI"), @selector(objectAtIndexedSubscript:));
    Method my_objectAtIndexedSubscriptMethod = class_getInstanceMethod(self, @selector(yxc_objectAtIndexedSubscript:));
    method_exchangeImplementations(system_objectAtIndexedSubscriptMethod, my_objectAtIndexedSubscriptMethod);

这时候我们重新编译运行,这时候程序没有在crash了。

最后附上,替换系统一些方法的代码:

+ (void)load {
    
    Method system_arrayWithObjectsCountMethod = class_getClassMethod(self, @selector(arrayWithObjects:count:));
    Method my_arrayWithObjectsCountMethod = class_getClassMethod(self, @selector(yxc_arrayWithObjects:count:));
    method_exchangeImplementations(system_arrayWithObjectsCountMethod, my_arrayWithObjectsCountMethod);
    
    Method system_objectAtIndexedSubscriptMethod = class_getInstanceMethod(NSClassFromString(@"__NSArrayI"), @selector(objectAtIndexedSubscript:));
    Method my_objectAtIndexedSubscriptMethod = class_getInstanceMethod(self, @selector(yxc_objectAtIndexedSubscript:));
    method_exchangeImplementations(system_objectAtIndexedSubscriptMethod, my_objectAtIndexedSubscriptMethod);

    Method system_objectAtIndexMethod = class_getInstanceMethod(NSClassFromString(@"__NSArrayI"), @selector(objectAtIndex:));
    Method my_objectAtIndexMethod = class_getInstanceMethod(self, @selector(yxc_objectAtIndex:));
    method_exchangeImplementations(system_objectAtIndexMethod, my_objectAtIndexMethod);
    
    Method system_addObjectMethod = class_getInstanceMethod(NSClassFromString(@"__NSArrayM"), @selector(addObject:));
    Method my_addObjectMethod = class_getInstanceMethod(self, @selector(yxc_addObject:));
    method_exchangeImplementations(system_addObjectMethod, my_addObjectMethod);
    
    Method system_insertObjectAtIndexMethod = class_getInstanceMethod(NSClassFromString(@"__NSArrayM"), @selector(insertObject:atIndex:));
    Method my_insertObjectAtIndexMethod = class_getInstanceMethod(self, @selector(yxc_insertObject:atIndex:));
    method_exchangeImplementations(system_insertObjectAtIndexMethod, my_insertObjectAtIndexMethod);
    
    Method system_removeObjectAtIndexMethod = class_getInstanceMethod(NSClassFromString(@"__NSArrayM"), @selector(removeObjectAtIndex:));
    Method my_removeObjectAtIndexMethod = class_getInstanceMethod(self, @selector(yxc_removeObjectAtIndex:));
    method_exchangeImplementations(system_removeObjectAtIndexMethod, my_removeObjectAtIndexMethod);

    // 此处是可变数组的取值方法替换
    Method system_objectAtIndexedSubscriptMethod1 = class_getInstanceMethod(NSClassFromString(@"__NSArrayM"), @selector(objectAtIndexedSubscript:));
    Method my_objectAtIndexedSubscriptMethod1 = class_getInstanceMethod(self, @selector(yxc_objectAtIndexedSubscript1:));
    method_exchangeImplementations(system_objectAtIndexedSubscriptMethod1, my_objectAtIndexedSubscriptMethod1);
}

/**
 @[] 字面量初始化调用方法

 @param objects 对象
 @param cnt 数组个数
 @return 数组
 */
+ (instancetype)yxc_arrayWithObjects:(id  _Nonnull const [])objects count:(NSUInteger)cnt {
    
    NSMutableArray *objectArray = [NSMutableArray array];
    
    for (int i = 0; i < cnt; i++) {
        id object = objects[i];
        if (object && ![object isKindOfClass:[NSNull class]]) {
            [objectArray addObject:object];
        }
    }
    
    return [NSArray arrayWithArray:objectArray];
}

/**
 数组添加一个对象
 */
- (void)yxc_addObject:(id)anObject {
    
    if (!anObject) return;
    [self yxc_addObject:anObject];
}

/**
 数组插入一个对象

 @param anObject 对象
 @param index 待插入的下标
 */
- (void)yxc_insertObject:(id)anObject atIndex:(NSUInteger)index {
    
    if (!anObject) return;
    if (index > self.count) return; // 数组可以插入下标为0这个位置,如果此处 >= 会有问题
    
    [self yxc_insertObject:anObject atIndex:index];
}

/**
 根据下标移除某个对象

 @param index 需要移除的下标
 */
- (void)yxc_removeObjectAtIndex:(NSUInteger)index {
    
    if (index >= self.count) return;
    
    [self yxc_removeObjectAtIndex:index];
}

/**
 通过 index 获取对象

 @param index 数组下标
 */
- (id)yxc_objectAtIndex:(NSUInteger)index {
    
    if (index >= self.count) return nil;
    
    return [self yxc_objectAtIndex:index];
}

/**
 @[] 形式获取数组对象

 @param idx 数组下标
 */
- (id)yxc_objectAtIndexedSubscript:(NSUInteger)idx {
    
    if (idx >= self.count) return nil;
    
    return [self objectAtIndex:idx];
}

/**
 @[] 形式获取数组对象
 
 @param idx 数组下标
 */
- (id)yxc_objectAtIndexedSubscript1:(NSUInteger)idx {
    
    if (idx >= self.count) return nil;
    
    return [self objectAtIndex:idx];
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,504评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,434评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,089评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,378评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,472评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,506评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,519评论 3 413
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,292评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,738评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,022评论 2 329
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,194评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,873评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,536评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,162评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,413评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,075评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,080评论 2 352

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,698评论 0 9
  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,094评论 1 32
  • 没学非暴前,先生开车送我去学校,等的时间长了会生气,我会害怕生气,担心迟到,下车时我理所当然的走了,一句话也没有,...
    弓建萍阅读 145评论 0 0
  • 训练营这几天的活动,很快就经历完了,回头一看竟然已经过了7天, 时间过得好快,好像也没做什么。这段时间事特忙,不是...
    再向虎山行之路阅读 169评论 0 0
  • 一、springmvc的执行流程: 架构图: 具体执行步骤如下: 1、 首先用户发送请求————>前端控制器,前端...
    聂叼叼阅读 507评论 0 0