首页 > 解决方案 > 具有通用(可选)用户默认值的属性包装器

问题描述

适当参考:

https://github.com/apple/swift-evolution/blob/master/proposals/0258-property-wrappers.md#user-defaults

我们已经开始为 UserDefaults 使用属性包装器,它可以与非可选属性无缝协作。

但是,设置 nil 的可选属性会崩溃:

[用户默认值] 尝试将非属性列表对象设置为键“someKeyThatWeSet”的 NSUserDefaults/CFPreferences 值

由于未捕获的异常“NSInvalidArgumentException”而终止应用程序,原因:“尝试为键“someKeyThatWeSet”插入非属性列表对象 null

下面的代码可以直接在 Playground 上测试:

@propertyWrapper
struct C2AppProperty<T> {
    let key: String
    let defaultValue: T

    init(_ key: String, defaultValue: T) {
        self.key = key
        self.defaultValue = defaultValue
    }

    var wrappedValue: T {
        get {
            return UserDefaults.standard.object(forKey: key) as? T ?? defaultValue
        }
        set {
            UserDefaults.standard.set(newValue, forKey: key)
        }
    }
}

struct C2User {
    @C2AppProperty("userID", defaultValue: nil)
    public static var publicUserID: String?
}

print(C2User.publicUserID)
C2User.publicUserID = "edusta"
print(C2User.publicUserID)
C2User.publicUserID = nil
print(C2User.publicUserID)

预期的:

可选<"edusta">

成立:

可选<"edusta">

libc++abi.dylib:以 NSException 类型的未捕获异常终止

到目前为止我已经尝试过:

set {
    // Comparing non-optional value of type 'T' to nil always returns false.
    if newValue == nil {
        UserDefaults.standard.removeObject(forKey: combinedKey)
    } else {
        UserDefaults.standard.set(newValue, forKey: combinedKey)
    }
}

这里需要什么样的检查才能捕捉到 newValue 是nil什么?还是一个Optional<nil>

标签: iosswiftgenericsnsuserdefaults

解决方案


这段代码对我有用:

@propertyWrapper
struct UserDefault<T> {
    private let key: String
    private let defaultValue: T
    private let userDefaults: UserDefaults

    init(_ key: String, defaultValue: T, userDefaults: UserDefaults = .standard) {
        self.key = key
        self.defaultValue = defaultValue
        self.userDefaults = userDefaults
    }

    var wrappedValue: T {
        get {
            guard let value = userDefaults.object(forKey: key) else {
                return defaultValue
            }

            return value as? T ?? defaultValue
        }
        set {
            if let value = newValue as? OptionalProtocol, value.isNil() {
                userDefaults.removeObject(forKey: key)
            } else {
                userDefaults.set(newValue, forKey: key)
            }
        }
    }
}

fileprivate protocol OptionalProtocol {
    func isNil() -> Bool
}

extension Optional : OptionalProtocol {
    func isNil() -> Bool {
        return self == nil
    }
}

推荐阅读