使用 Swift 将可编码结构保存到 UserDefaults

F. *_*uta 3 swift userdefaults codable

我正在尝试对结构进行编码

struct Configuration : Encodable, Decodable {
    private enum CodingKeys : String, CodingKey {
        case title = "title"
        case contents = "contents"
    }
    var title : String?
    var contents: [[Int]]?
}
Run Code Online (Sandbox Code Playgroud)

转换成 JSON 以存储在 UserDefaults.standard 的本地键中。我有以下代码:

let jsonString = Configuration(title: nameField.text, contents: newContents)
let info = ["row" as String: jsonString as Configuration]
print("jsonString = \(jsonString)")
//trying to save object
let defaults = UserDefaults.standard
let recode = try! JSONEncoder().encode(jsonString)
defaults.set(recode, forKey: "simulationConfiguration")
//end of saving local
Run Code Online (Sandbox Code Playgroud)

打印返回:

jsonString = Configuration(title: Optional("config"), contents: Optional([[4, 5], [5, 5], [6, 5]]))
Run Code Online (Sandbox Code Playgroud)

所以我相信我正确地创建了对象。但是,当我下次运行模拟器时尝试检索密钥时,我什么也得不到。我把以下内容放在 AppDelegate 中,它总是返回 No Config。

let defaults = UserDefaults.standard
        let config = defaults.string(forKey: "simulationConfiguration") ?? "No Config"
        print("from app delegate = \(config.description)")
Run Code Online (Sandbox Code Playgroud)

有任何想法吗?谢谢

Luc*_*tti 12

在这里您正在保存一个Data值(这是正确的)

defaults.set(recode, forKey: "simulationConfiguration")
Run Code Online (Sandbox Code Playgroud)

但在这里你正在阅读 String

defaults.string(forKey: "simulationConfiguration")
Run Code Online (Sandbox Code Playgroud)

您无法保存Data、阅读String并期望它起作用。

让我们修复您的代码

首先,您不需要手动指定编码密钥。所以你的结构变成了这个

struct Configuration : Codable {
    var title : String?
    var contents: [[Int]]?
}
Run Code Online (Sandbox Code Playgroud)

保存

现在这是保存它的代码

let configuration = Configuration(title: "test title", contents: [[1, 2, 3]])
if let data = try? JSONEncoder().encode(configuration) {
    UserDefaults.standard.set(data, forKey: "simulationConfiguration")
}
Run Code Online (Sandbox Code Playgroud)

加载中

这是阅读它的代码

if
    let data = UserDefaults.standard.value(forKey: "simulationConfiguration") as? Data,
    let configuration = try? JSONDecoder().decode(Configuration.self, from: data) {
    print(configuration)
}
Run Code Online (Sandbox Code Playgroud)