ios - 尝试在 Swift 3 中保存自定义对象时尝试插入非属性列表对象

标签 ios swift swift3 nsuserdefaults nscoding

我有一个符合 NSCoding 协议(protocol)的简单对象。

import Foundation

class JobCategory: NSObject, NSCoding {
    var id: Int
    var name: String
    var URLString: String

    init(id: Int, name: String, URLString: String) {
        self.id = id
        self.name = name
        self.URLString = URLString
    }

    // MARK: - NSCoding
    required init(coder aDecoder: NSCoder) {
        id = aDecoder.decodeObject(forKey: "id") as? Int ?? aDecoder.decodeInteger(forKey: "id")
        name = aDecoder.decodeObject(forKey: "name") as! String
        URLString = aDecoder.decodeObject(forKey: "URLString") as! String
    }

    func encode(with aCoder: NSCoder) {
        aCoder.encode(id, forKey: "id")
        aCoder.encode(name, forKey: "name")
        aCoder.encode(URLString, forKey: "URLString")
    }
}

我尝试将其实例保存在 UserDefaults 中,但它始终失败并出现以下错误。

Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: 'Attempt to insert non-property list object for key jobCategory'

这是我在 UserDefaults 中保存的代码。

enum UserDefaultsKeys: String {
    case jobCategory
}

class ViewController: UIViewController {

    @IBAction func didTapSaveButton(_ sender: UIButton) {
        let category = JobCategory(id: 1, name: "Test Category", URLString: "http://www.example-job.com")

        let userDefaults = UserDefaults.standard
        userDefaults.set(category, forKey: UserDefaultsKeys.jobCategory.rawValue)
        userDefaults.synchronize()
    }
}

我用普通字符串替换了 key 的枚举值,但仍然出现相同的错误。知道是什么原因造成的吗?

最佳答案

您需要使用 JSONEncoder 从您的 JobCategory 模型创建 Data 实例并将该 Data 实例存储在 UserDefaults 中,然后使用 JSONDecoder 进行解码.

struct JobCategory: Codable {
    let id: Int
    let name: String
}

// To store in UserDefaults
if let encoded = try? JSONEncoder().encode(category) {
    UserDefaults.standard.set(encoded, forKey: UserDefaultsKeys.jobCategory.rawValue)
}

// Retrieve from UserDefaults
if let data = UserDefaults.standard.object(forKey: UserDefaultsKeys.jobCategory.rawValue) as? Data, 
   let category = try? JSONDecoder().decode(JobCategory.self, from: data) {
     print(category.name)
}

旧答案

您需要使用 archivedData(withRootObject:)JobCategory 实例创建 Data 实例并将该 Data 实例存储在 UserDefaults 中,然后使用 unarchiveTopLevelObjectWithData(_:) 取消存档,所以尝试这样。

用于在UserDefaults中存储数据

let category = JobCategory(id: 1, name: "Test Category", URLString: "http://www.example-job.com")
let encodedData = NSKeyedArchiver.archivedData(withRootObject: category, requiringSecureCoding: false)
let userDefaults = UserDefaults.standard
userDefaults.set(encodedData, forKey: UserDefaultsKeys.jobCategory.rawValue)

用于从UserDefaults检索数据

let decoded  = UserDefaults.standard.object(forKey: UserDefaultsKeys.jobCategory.rawValue) as! Data
let decodedTeams = NSKeyedUnarchiver.unarchiveTopLevelObjectWithData(decoded) as! JobCategory
print(decodedTeams.name)

关于ios - 尝试在 Swift 3 中保存自定义对象时尝试插入非属性列表对象,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57377293/

相关文章:

swift - 通用完成作为非通用通过

ios - 为什么我的 subview 没有添加到我的 ScrollView 中?

ios - iPad 中的 Google 登录用户代理问题

ios - 无法在ios swift中将文件保存到本地存储

ios - 如何正确地将对象从一个 Realm 对象复制到另一个对象

ios - 设置 Spotify API 时,无法将类型 'ViewController' 的值分配给类型 'SPTAudioStreamingDelegate!'

ios - 以小于帧大小的增量分页 UIScrollView

ios - 我正在尝试连接阿拉伯语和英语文本

ios - 如何快速获取日期的时间

IOS 相当于 Android 选择器