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/57770522/

相关文章:

ios - - (void)chatDidReceiveMessage :(QBChatMessage *)message not working

iOS 9 运行时 nop 死亡坑

ios - 无法将类型 'CGPoint!' 的值转换为预期的参数类型 'UnsafeMutablePointer<CGPoint>'

ios - 将 "Text View"中的文本分配给变量会使应用程序崩溃

javascript - 当用户点击 x 按钮时,iOS6 上的 add2home.js 不会关闭

ios - 我应该如何访问存储在包含容器 View 的 View Controller 中的 IBOutlet? ( swift 4)

ios - 尝试在 Swift 中追加字典

arrays - 将新数组返回/传递给函数(Swift 5)

ios - 如何在布局中设置填充宽度和自定义高度

ios - Swift 3.0 中的 POST 请求