UserDefault property wrapper not saving values iOS versions below iOS 13(UserDefault属性包装不保存值低于iOS 13的iOS版本)
问题描述
我正在使用属性包装来保存我的用户默认值。在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(或字符串等)这样的简单值虽然可以作为可编码结构的属性编码(例如),但不能对本身进行JSON编码。错误(您要丢弃的错误)对此非常清楚:
顶级布尔编码为数字JSON片段。
要查看此内容,只需运行以下代码:
do {
_ = try JSONEncoder().encode(false)
print("succeeded")
} catch {
print(error)
}
在iOS 12上,我们得到错误。在iOS 13上,我们得到"succeeded"
。
但如果我们包装Bool(或字符串等)在可编写代码的结构中,一切正常:
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编码为属性列表。您仍然需要包装器方法。
这篇关于UserDefault属性包装不保存值低于iOS 13的iOS版本的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持编程学习网!
本文标题为:UserDefault属性包装不保存值低于iOS 13的iOS版本
基础教程推荐
- 如何让对象对 Cocos2D 中的触摸做出反应? 2022-01-01
- 在 gmail 中为 ios 应用程序检索朋友的朋友 2022-01-01
- Kivy Buildozer 无法构建 apk,命令失败:./distribute.sh -m “kivy"d 2022-01-01
- android 应用程序已发布,但在 google play 中找不到 2022-01-01
- 当从同一个组件调用时,两个 IBAction 触发的顺序是什么? 2022-01-01
- 如何在 UIImageView 中异步加载图像? 2022-01-01
- Android:对话框关闭而不调用关闭 2022-01-01
- 如何在没有IB的情况下将2个按钮添加到右侧的UINavigationbar? 2022-01-01
- UIWebView 委托方法 shouldStartLoadWithRequest:在 WKWebView 中等效? 2022-01-01
- 如何在 iPhone 上显示来自 API 的 HTML 文本? 2022-01-01