iOS崩溃大扫除——null崩溃NullSafe

-[NSNull objectForKeyedSubscript:]: unrecognized selector sent to instance 0xcf3238
-[NSNull length]: unrecognized selector sent to instance 0x388a4a70

什么场景下回遇到null呢?

nil/Nil/null/NSNull全解
1、解析 JSON 数据时。因为在后端数据库里面,默认数据为 null ,如果修过数据后,又删除,那么数据库会自动补充上 null 。在接口返回数据时,就会把 NSNull 传给我们,解析出来就是 null 空对象。后端也可以做些调整,修改默认数据的方式:在创建表的时候,添加上 'not null default' 。
2、项目中为了向字典、数组、集合等存入空值,会使用到 NSNull ,在读取时,会读取到 null 。

当我们给一个 null( NSNull 对象)发送消息的话,很大可能会直接Crash( null 是有内存的),而发送给nil的话,是不会崩溃的。

解决方案

1、对可能出现空的字段进行非空判断

会造成 Crash 的字段解析成的对象是 NSNull 类型的,所以可以直接判断是不是此类型,这样的字段会比较多所以也会比较繁琐,也容易遗漏,但是也比较保险。

if (![object isKindOfClass:[NSNull class]]){
.....
}

2、字符串匹配, 替换 null 为 为空字符 ""

在获取到服务器返回的 JSON 时,返回结果时 string 对象,于是就先替换 null 为 为空字符"",然后再解析可避免 Crash 问题,这种方法比较巧妙,但若服务器的数据不太整洁也会有一定的问题。

json = [jsonStr  stringByReplacingOccurrencesOfString:@":null" withString:@":\"\""];

3、解析数据时把 NSNull 类型的值替换成 nil

使用宏定义。

#define VerifyValue(value)\
({id tmp;\
if ([value isKindOfClass:[NSNull class]])\
tmp = nil;\
else\
tmp = value;\
tmp;\
})\

在解析数据时,把接收到的 NSNull 类型的值替换成 nil 。

contact.contactPhone = VerifyValue(contactDic[@"send_ContactPhone"]);

4、使用 AFNetworking 提供的方法

如果是使用 AFNetworking 这个库做网络请求的话,可以用以下代码,自动帮你去掉空值

self.removesKeysWithNullValues = YES;

5、第三方库 NullSafe

NullSafe 是一个 Category ,在运行时操作,把这个讨厌的空值置为 nil ,而 nil 是安全的,可以向 nil 对象发送任何 message 而不会 Crash ,其原理便是使用消息转发机制实现。只需要将文件 nullSafe.m 导入工程便可以,无需引用头文件等。 消息转发原理浅析

附: NullSafe 的工作原理

当我们给一个 NSNull 对象发送消息的话,可能会崩溃( null 是有内存的),而发送给 nil 的话,是不会崩溃的。发送给 NSNull 而 NSNull 又无法处理的消息要经过如下几步做转发处理:

  1. 创建一个方法缓存,这个缓存会缓存项目中类的所有类名。
  2. 遍历缓存,寻找是否已经有可以执行此方法的类。
  3. 如果有的话,返回这个 NSMethodSignature 。
  4. 如果没有的话,返回 nil , 接下来会走 forwardInvocation:方法。
  5. [invocation invokeWithTarget:nil]; 将消息转发给 nil。

那么,如何判断NSNull无法处理这个消息呢,在OC中,系统如果对某个实例发送消息之后,它(及其父类)无法处理(比如,没有这个方法等),系统就会发送 methodSignatureForSelector 消息,如果这个方法返回非空,那么就去执行返回的方法,如果为 nil, 则发送 forwardInvocation 消息。这样就完成整个转发链了。

你可能感兴趣的:(iOS崩溃大扫除——null崩溃NullSafe)