动态密钥 属性 包装器

Dynamic key property wrapper

我编写了一个简单的 属性 包装器,可以轻松地在 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。有没有办法解决这个问题,这样我就可以轻松地使用带有动态密钥的 属性 包装器?

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

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)
}

两种解决方案都没有使用 UserDefault@propertyWrapper 方面,但在这里似乎没有必要——它仍然会产生相同的 type/functionality.