动态键属性包装器

问题描述

我编写了一个简单的属性包装器,可以在 UserDefaults 中轻松获取和设置用户首选项。

@propertyWrapper
struct UserDefault<Value> {
    let key: String
    let defaultValue: Value

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

// Example usage
@UserDefault(key: "com.example.flag",defaultValue: false)
var trackingEnabled: Bool

在这适用于静态首选项,但假设我要支持多个用户,每个用户都可以存储他们的设置。

struct Person {
    let id: UUID
    let name: String

    @UserDefault(key: "com.example.person.\(id)",defaultValue: false)
    var trackingEnabled: Bool
}

这给了我警告 Cannot use instance member 'id' within property initializer; property initializers run before 'self' is available。有没有办法解决这个问题,以便我可以轻松地将此属性包装器与动态密钥一起使用?

解决方法

为了在 id 的初始化中使用 trackingEnabled 的值,您可以为您的结构使用自定义初始化程序:

struct Person {
    let id: UUID
    let name: String

    var trackingEnabled: UserDefault<Bool>
    
    init(id: UUID,name: String) {
        self.id = id
        self.name = name
        trackingEnabled = UserDefault(key: "com.example.person.\(id)",defaultValue: false)
    }
}

或者,您可以将 trackingEnabled 定义为 lazy,以便它在初始化时可以访问 id 参数:

struct Person {
    let id: UUID
    let name: String

    lazy var trackingEnabled = UserDefault(key: "com.example.person.\(id)",defaultValue: false)
}

两个解决方案都没有使用 @propertyWrapperUserDefault 方面,但在这里似乎没有必要——它仍然产生相同的类型/功能。