NSKeyedArchiver:强制转换返回零的数据-Swift

时间:2019-02-12 18:14:48

标签: swift nscoding nskeyedarchiver

好吧,在这里研究了几个类似的主题,按照建议进行了所有操作,但是当尝试将解码的对象转换为我的类型时,我的计算属性“ previousUserData”返回nil。怎么了?

@objc class PreviousUserData: NSObject, NSCoding {

var name: String
var phone: String
var email: String

func encode(with aCoder: NSCoder) {
    aCoder.encode(name, forKey: "name")
    aCoder.encode(phone, forKey: "phone")
    aCoder.encode(email, forKey: "email")
}

required convenience init?(coder aDecoder: NSCoder) {
    guard
        let name = aDecoder.decodeObject(forKey: "name") as? String,
        let phone = aDecoder.decodeObject(forKey: "phone") as? String,
        let email = aDecoder.decodeObject(forKey: "email") as? String
        else {
            return nil
    }
    self.init(name: name, phone: phone, email: email)
}

init(name: String, phone: String, email: String) {
    self.name = name
    self.phone = phone
    self.email = email
}
}

unarchived返回nil,但是键“ userdata”的数据存在

    var previousUserData: PreviousUserData? {
    get {
        if let object = UserDefaults.standard.object(forKey: "userdata") as? Data {
            let unarchived = NSKeyedUnarchiver.unarchiveObject(with: object) as? PreviousUserData
            return unarchived
        }
        return nil
    }
    set {
        let encodedData: Data = NSKeyedArchiver.archivedData(withRootObject: previousUserData as Any)
        UserDefaults.standard.setValue(encodedData, forKey: "userdata")
    }
    }

1 个答案:

答案 0 :(得分:2)

实际上,由于设置程序错误,您无法获得有效数据。您必须保存newValue而不是previousUserData

这是一个经过优化的版本

var previousUserData: PreviousUserData? {
    get {
        guard let data = UserDefaults.standard.data(forKey: "userdata") else { return nil }
        return NSKeyedUnarchiver.unarchiveObject(with: data) as? PreviousUserData
    }
    set {
        guard let newValue = newValue else { return }
        let encodedData = NSKeyedArchiver.archivedData(withRootObject: newValue)
        UserDefaults.standard.set(encodedData, forKey: "userdata")
    }
}

NSCoding很重。在这种情况下,我建议使用Codable将数据序列化为JSON或“属性列表”。它消除了@objcclassNSObject,并将整个代码减少为

struct PreviousUserData : Codable {
    var name: String
    var phone: String
    var email: String
}

var previousUserData: PreviousUserData? {
    get {
        guard let data = UserDefaults.standard.data(forKey: "userdata") else { return nil }
        return try? JSONDecoder().decode(PreviousUserData.self, from: data)
    }
    set {
        guard let newValue = newValue, let encodedData = try? JSONEncoder().encode(newValue) else { return }
        UserDefaults.standard.set(encodedData, forKey: "userdata")
    }
}