iOSiOS开发实用技巧技术重塑

判断NSString为空容易犯的错

2017-03-09  本文已影响1324人  纸简书生

判断字符串为空是一件简单不能再简单的事情。昨天有同学在这件简单的事情上栽更头了。看似简单其实就能看出内在的功力。

看似没有问题

判断字符串是否为空使用频率非常高,所以一般做法是为NSString创建一个分类,然后直接通过分类来调用。比如:

@interface NSString (Util)
- (BOOL)isBlankString:(NSString *)str;
@end

判断字符串是否为空的情况,不仅仅是[string isEqualToString:@""]这么简单(也可能跟业务有关),我平时判断的逻辑如下:

  1. 是否为nil
  2. 是否是NSNull
  3. 是否去掉空格之后长度为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 发送消息是完全有效的——只是在运行时不会有任何作用

主要分为如下几个类型:

大致分为返回值对象、指针、结构体、这三种。

那么根据上面的结论当调用[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)) {
     // 为空处理                     
}
上一篇下一篇

猜你喜欢

热点阅读