How to allow NSMutableDictionary to accept 'nil' values?

后端 未结 3 2105
遇见更好的自我
遇见更好的自我 2021-02-12 16:01

I have this statement:

 [custData setObject: [rs stringForColumnIndex:2]  forKey: @\"email\"];

where [rs stringForColumnIndex:2] o

相关标签:
3条回答
  • 2021-02-12 16:45

    It is not possible with a pure NSMutableDictionary, and in most cases you want to convert nil values into [NSNull null] or just omit them from the dictionary. Sometimes (very seldom), though, it is convenient to allow nil values, and in those cases you can use CFMutableDictionary with custom callbacks.

    If you go this way, I recommend that you use CoreFoundation API for all accesses, e.g. CFDictionarySetValue and CFDictionaryGetValue.

    However, if you know what you're doing, you can use toll-free bridging and cast that CFMutableDictionary to NSMutableDictionary or NSDictionary. This may be useful if you have a bunch of helpers that accept NSDictionary, and you want to use them on your modified nil-capable dictionary. (Of course, make sure that the helpers aren't surprised by nil values.)

    If you do the bridging, note that:

    1) NSMutableDictionary setter raises errors on nil values before bridging, so you need to use CFDictionarySetValue to set values that are potentially nil.

    2) technically, we're violating a contract of NSMutableDictionary here, and things may break (e.g. in future OS updates)

    3) a lot of code will be very surprised to find nil values in a dictionary; you should only pass the bridged frankendictionaries to the code that you control

    See ridiculousfish's post on toll-free bridging for an explanation of why a bridged CFDictionary behaves differently from NSDictionary.

    Example:

    #import <Foundation/Foundation.h>
    
    const void *NullSafeRetain(CFAllocatorRef allocator, const void *value) {
        return value ? CFRetain(value) : NULL;
    }
    void NullSafeRelease(CFAllocatorRef allocator, const void *value) {
        if (value)
            CFRelease(value);
    }
    const CFDictionaryValueCallBacks kDictionaryValueCallBacksAllowingNULL = {
        .version = 0,
        .retain = NullSafeRetain,
        .release = NullSafeRelease,
        .copyDescription = CFCopyDescription,
        .equal = CFEqual,
    };
    
    int main(int argc, const char * argv[])
    {
        @autoreleasepool {
            CFMutableDictionaryRef cfdictionary = CFDictionaryCreateMutable(NULL, 0, &kCFTypeDictionaryKeyCallBacks, &kDictionaryValueCallBacksAllowingNULL);
            CFDictionarySetValue(cfdictionary, @"foo", @"bar");
            CFDictionarySetValue(cfdictionary, @"boz", nil);
    
            NSMutableDictionary *dictionary = CFBridgingRelease(cfdictionary);
            NSLog(@"dictionary[foo] = %@", dictionary[@"foo"]);
            NSLog(@"dictionary[foo] = %@", dictionary[[@"fo" stringByAppendingString:@"o"]]);
            NSLog(@"dictionary[boz] = %@", dictionary[@"boz"]);
            NSLog(@"dictionary = %@", dictionary);
            NSLog(@"(dictionary isEqualTo: dictionary) = %d", [dictionary isEqualToDictionary:dictionary]);
        }
        return 0;
    }
    

    outputs:

    dictionary[foo] = bar
    dictionary[foo] = bar
    dictionary[boz] = (null)
    dictionary = {
        boz = (null);
        foo = bar;
    }
    (dictionary isEqualTo: dictionary) = 1
    
    0 讨论(0)
  • 2021-02-12 16:46

    I needed to set a NSDictionary value to one that may or may not be set yet from NSUserDefaults.

    What I did was wrap the values in a stringwithFormat call. Both values are not yet set so start as null. When I run without the stringwithFormat call the app crashes. So I did this and in my situation worked.

    -(NSDictionary*)userDetailsDict{
    NSDictionary* userDetails = @{
                                  @"userLine":[NSString stringWithFormat:@"%@",[[NSUserDefaults standardUserDefaults]stringForKey:kSelectedLine] ],
                                  @"userDepot":[NSString stringWithFormat:@"%@",[[NSUserDefaults standardUserDefaults]stringForKey:@"kSelected Duty Book"]]
                                  };
    
    return userDetails;
    }
    
    0 讨论(0)
  • 2021-02-12 16:52

    There is a non-nil object called NSNull that is built specifically to represent nils in situations where "plain" nil is not acceptable. If you replace your nils with [NSNull null] object, NSDictionary will take them. You would need to check for NSNull on the way out, though.

    Note that this is important only when you must differentiate between a value not being set and a value being set to nil. If your code is such that it can interpret a missing value as nil, you do not need to use NSNull at all.

    0 讨论(0)
提交回复
热议问题