美文网首页iOSiOS开发实用技巧技术重塑
判断NSString为空容易犯的错

判断NSString为空容易犯的错

作者: 纸简书生 | 来源:发表于2017-03-09 14:51 被阅读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 发送消息是完全有效的——只是在运行时不会有任何作用

    主要分为如下几个类型:

    • 如果一个方法返回值是一个对象,那么发送给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)) {
         // 为空处理                     
    }
    

    相关文章

      网友评论

      • inoryshu:不觉得前后判断的方法有问题吗?(BOOL)isBlankString:(NSString *)str 这个方法无论是类方法还是-方法,因为你是把判断的str直接方法传过来的,所以判断是否为nil是根本没有问题的。然而 BOOL isBlank = [str isBlankString];这个类方法才会出现str为nil,方法调用不了的问题。
      • 谢谢生活:为什么str为空回犯规,nil,不进入函数返回yes呢,还有宏为什么就能解决问题,有什么好处
        liangdahong:@谢谢生活 为nil 时 无法调起方法,这里直接返回0了 使用判断错误,用宏定义是调的类方法、肯定会走方法 那么就不会出现刚才的情况
      • 春暖花已开:不错!

      本文标题:判断NSString为空容易犯的错

      本文链接:https://www.haomeiwen.com/subject/fkgwgttx.html