UserDefault 属性 包装器不保存值 iOS 低于 iOS 13 的版本
UserDefault property wrapper not saving values iOS versions below iOS 13
我正在使用 属性 包装器来保存我的用户默认值。在 iOS 13 台设备上,此解决方案效果很好。但是在 iOS 11 和 iOS 12 上,这些值没有保存到用户默认值中。我读到 属性 包装器是向后兼容的,所以我不知道为什么这不适用于旧的 iOS 版本。
这是 属性 包装器:
@propertyWrapper
struct UserDefaultWrapper<T: Codable> {
private let key: String
private let defaultValue: T
init(key: String, defaultValue: T) {
self.key = key
self.defaultValue = defaultValue
}
var wrappedValue: T {
get {
guard let data = UserDefaults.standard.object(forKey: key) as? Data else {
// Return defaultValue when no data in UserDefaults
return defaultValue
}
// Convert data to the desire data type
let value = try? JSONDecoder().decode(T.self, from: data)
return value ?? defaultValue
}
set {
// Convert newValue to data
let data = try? JSONEncoder().encode(newValue)
UserDefaults.standard.set(data, forKey: key)
UserDefaults.standard.synchronize()
}
}
}
struct UserDefault {
@UserDefaultWrapper(key: "userIsSignedIn", defaultValue: false)
static var isSignedIn: Bool
}
然后我可以像这样设置值:
UserDefault.isSignedIn = true
我使用的 属性 包装器错了吗?是否还有其他人 运行 对旧 iOS 版本的 属性 包装器存在问题?
与 属性 包装器无关!问题是,在 iOS 12 及之前,一个简单的值,如 Bool(或 String 等),尽管 Codable 是 Codable 的 属性结构(例如),不能 自身 被 JSON 编码。这个错误(你扔掉的)很清楚:
Top-level Bool encoded as number JSON fragment.
要查看此内容,只需 运行 此代码:
do {
_ = try JSONEncoder().encode(false)
print("succeeded")
} catch {
print(error)
}
在 iOS 12,我们收到错误。在 iOS 13,我们得到 "succeeded"
。
但是如果我们将 Bool(或 String 等)包装在 Codable 结构中,一切都很好:
struct S : Codable { let prop : Bool }
do {
_ = try JSONEncoder().encode(S(prop:false))
print("succeeded")
} catch {
print(error)
}
在 iOS 12 和 iOS 13.
上工作正常
这个事实提出了一个解决方案!重新定义您的 属性 包装器,以便将其值包装在通用包装器结构中:
struct UserDefaultWrapper<T: Codable> {
struct Wrapper<T> : Codable where T : Codable {
let wrapped : T
}
private let key: String
private let defaultValue: T
init(key: String, defaultValue: T) {
self.key = key
self.defaultValue = defaultValue
}
var wrappedValue: T {
get {
guard let data = UserDefaults.standard.object(forKey: key) as? Data
else { return defaultValue }
let value = try? JSONDecoder().decode(Wrapper<T>.self, from: data)
return value?.wrapped ?? defaultValue
}
set {
do {
let data = try JSONEncoder().encode(Wrapper(wrapped:newValue))
UserDefaults.standard.set(data, forKey: key)
} catch {
print(error)
}
}
}
}
现在适用于 iOS 12 和 iOS 13.
顺便说一下,实际上我认为您最好保存为 属性 列表而不是 JSON。但这对一般的问题没有影响。您也不能将纯 Bool 编码为 属性 列表。您仍然需要 Wrapper 方法。
我正在使用 属性 包装器来保存我的用户默认值。在 iOS 13 台设备上,此解决方案效果很好。但是在 iOS 11 和 iOS 12 上,这些值没有保存到用户默认值中。我读到 属性 包装器是向后兼容的,所以我不知道为什么这不适用于旧的 iOS 版本。
这是 属性 包装器:
@propertyWrapper
struct UserDefaultWrapper<T: Codable> {
private let key: String
private let defaultValue: T
init(key: String, defaultValue: T) {
self.key = key
self.defaultValue = defaultValue
}
var wrappedValue: T {
get {
guard let data = UserDefaults.standard.object(forKey: key) as? Data else {
// Return defaultValue when no data in UserDefaults
return defaultValue
}
// Convert data to the desire data type
let value = try? JSONDecoder().decode(T.self, from: data)
return value ?? defaultValue
}
set {
// Convert newValue to data
let data = try? JSONEncoder().encode(newValue)
UserDefaults.standard.set(data, forKey: key)
UserDefaults.standard.synchronize()
}
}
}
struct UserDefault {
@UserDefaultWrapper(key: "userIsSignedIn", defaultValue: false)
static var isSignedIn: Bool
}
然后我可以像这样设置值:
UserDefault.isSignedIn = true
我使用的 属性 包装器错了吗?是否还有其他人 运行 对旧 iOS 版本的 属性 包装器存在问题?
与 属性 包装器无关!问题是,在 iOS 12 及之前,一个简单的值,如 Bool(或 String 等),尽管 Codable 是 Codable 的 属性结构(例如),不能 自身 被 JSON 编码。这个错误(你扔掉的)很清楚:
Top-level Bool encoded as number JSON fragment.
要查看此内容,只需 运行 此代码:
do {
_ = try JSONEncoder().encode(false)
print("succeeded")
} catch {
print(error)
}
在 iOS 12,我们收到错误。在 iOS 13,我们得到 "succeeded"
。
但是如果我们将 Bool(或 String 等)包装在 Codable 结构中,一切都很好:
struct S : Codable { let prop : Bool }
do {
_ = try JSONEncoder().encode(S(prop:false))
print("succeeded")
} catch {
print(error)
}
在 iOS 12 和 iOS 13.
上工作正常这个事实提出了一个解决方案!重新定义您的 属性 包装器,以便将其值包装在通用包装器结构中:
struct UserDefaultWrapper<T: Codable> {
struct Wrapper<T> : Codable where T : Codable {
let wrapped : T
}
private let key: String
private let defaultValue: T
init(key: String, defaultValue: T) {
self.key = key
self.defaultValue = defaultValue
}
var wrappedValue: T {
get {
guard let data = UserDefaults.standard.object(forKey: key) as? Data
else { return defaultValue }
let value = try? JSONDecoder().decode(Wrapper<T>.self, from: data)
return value?.wrapped ?? defaultValue
}
set {
do {
let data = try JSONEncoder().encode(Wrapper(wrapped:newValue))
UserDefaults.standard.set(data, forKey: key)
} catch {
print(error)
}
}
}
}
现在适用于 iOS 12 和 iOS 13.
顺便说一下,实际上我认为您最好保存为 属性 列表而不是 JSON。但这对一般的问题没有影响。您也不能将纯 Bool 编码为 属性 列表。您仍然需要 Wrapper 方法。