判断字符串为空是一件简单不能再简单的事情。昨天有同学在这件简单的事情上栽更头了。看似简单其实就能看出内在的功力。
看似没有问题
判断字符串是否为空使用频率非常高,所以一般做法是为NSString创建一个分类,然后直接通过分类来调用。比如:
@interface NSString (Util)
- (BOOL)isBlankString:(NSString *)str;
@end
判断字符串是否为空的情况,不仅仅是[string isEqualToString:@""]这么简单(也可能跟业务有关),我平时判断的逻辑如下:
- 是否为nil
- 是否是NSNull
- 是否去掉空格之后长度为0
通过如上几个步骤能够确定字符串是否为空。
.m文件如下:
@implementation NSString (Util)
- (BOOL)isBlankString:(NSString *)str {
NSString *string = str;
if (string == nil || string == NULL) {
return YES;
}
if ([string isKindOfClass:[NSNull class]]) {
return YES;
}
if ([[string stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceCharacterSet]] length]==0) {
return YES;
}
return NO;
}
这一切都看似没问题,只是问题藏得有点深。比如当前字符串为nil的时候
问题重现
将上诉代码用验证,会发现isBlank为NO,也就是字符串不为空,????:
NSString *str = (网络解析出来的数据,解析结果为nil);
BOOL isBlank = [str isBlankString];
让我们来看看是哪里出了问题。哦!如果你知道objc是动态语言,每个方法在运行时会被动态转为消息发送,即:objc_msgSend(receiver, selector)的话那就应该知道原因了。
问题原因
先把结论总结了:
好多同学不知道:在 Objective-C 中向 nil 发送消息是完全有效的——只是在运行时不会有任何作用
主要分为如下几个类型:
- 如果一个方法返回值是一个对象,那么发送给nil的消息将返回0(nil)。例如:
Person * motherInlaw = [[aPerson spouse] mother];
- 如果 spouse 对象为 nil,那么发送给 nil 的消息 mother 也将返回 nil。
- 如果方法返回值为指针类型,其指针大小为小于或者等于sizeof(void*),float,double,long double 或者 long long 的整型标量,发送给 nil 的消息将返回0。
- 如果方法返回值为结构体,发送给 nil 的消息将返回0。结构体中各个字段的值将都是0。
- 如果方法的返回值不是上述提到的几种情况,那么发送给 nil 的消息的返回值将是未定义的。
大致分为返回值对象、指针、结构体、这三种。
那么根据上面的结论当调用[str isBlankString]
的时候,str的值为nil,str是一个对象,那么就会返回为nil。而nil。而nil对应的值为0,再对应到Bool上就是NO。所以一个本来为空的字符串就被判断为不为空了
关于nil/Nil/NULL/NSNull/的区别,详细请看这里
深入分析
为了理解这个问题,看一看objc的源代码(可能不是最新的版本)。
struct objc_class {
Class isa OBJC_ISA_AVAILABILITY; //isa指针指向Meta Class,因为Objc的类的本身也是一个Object,为了处理这个关系,runtime就创造了Meta Class,当给类发送[NSObject alloc]这样消息时,实际上是把这个消息发给了Class Object
#if !__OBJC2__
Class super_class OBJC2_UNAVAILABLE; // 父类
const char *name OBJC2_UNAVAILABLE; // 类名
long version OBJC2_UNAVAILABLE; // 类的版本信息,默认为0
long info OBJC2_UNAVAILABLE; // 类信息,供运行期使用的一些位标识
long instance_size OBJC2_UNAVAILABLE; // 该类的实例变量大小
struct objc_ivar_list *ivars OBJC2_UNAVAILABLE; // 该类的成员变量链表
struct objc_method_list **methodLists OBJC2_UNAVAILABLE; // 方法定义的链表
struct objc_cache *cache OBJC2_UNAVAILABLE; // 方法缓存,对象接到一个消息会根据isa指针查找消息对象,这时会在method Lists中遍历,如果cache了,常用的方法调用时就能够提高调用的效率。
struct objc_protocol_list *protocols OBJC2_UNAVAILABLE; // 协议链表
#endif
} OBJC2_UNAVAILABLE;
objc在向一个对象发送消息时,runtime库会根据对象的isa指针找到该对象实际所属的类,然后在该类中的方法列表以及其父类方法列表中寻找方法运行,然后在发送消息的时候,objc_msgSend方法不会返回值,所谓的返回内容都是具体调用时执行的。 那么,如果向一个nil对象发送消息,首先在寻找对象的isa指针时就是0地址返回了,所以不会出现任何错误。
改进
后来我建议他用类方法来判断是否为空,然后通过宏定义方式来快速调用。详细代码可以参考如下:
.h
+ (BOOL)isBlankString:(NSString *)str;
.m
+ (BOOL)isBlankString:(NSString *)str {
NSString *string = str;
if (string == nil || string == NULL) {
return YES;
}
if ([string isKindOfClass:[NSNull class]]) {
return YES;
}
if ([[string stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceCharacterSet]] length]==0) {
return YES;
}
return NO;
}
宏定义:
#pragma mark - NSString Macro
#define KIsBlankString(str) [NSString isBlankString:str]
使用:
if (KIsBlankString(avatar)) {
// 为空处理
}