NSKeyedUnarchiver 在检索自定义 class 时返回 nil
NSKeyedUnarchiver returning nil when retrieving custom class
在我的视图控制器中,我有一个数组中的自定义 class 存储。我使用 NSKeyedArchiver 来存储数据,但我使用的是折旧版本 NSKeyedUnarchiver.unarchiveObject(withFile: object.ArchiveURL.path) as? [object]
.
现在我决定更新为:
let data = try Data(contentsOf: filePath)
let object = try NSKeyedUnarchiver.unarchiveTopLevelObjectWithData(data)
这是我的保存功能:
let data = try NSKeyedArchiver.archivedData(withRootObject: object, requiringSecureCoding: false)
try data.write(to: filePath)
当我 return 从 Unarchiver 反对时,它 return 没有。我不确定为什么。
这是我的习惯 class:
class object: NSObject, NSCoding {
var name: String
var photo: UIImage
struct PropertyKey {
static let name = "name"
static let photo = "photo"
}
func encode(with aCoder: NSCoder) {
aCoder.encode(self.name, forKey: PropertyKey.name)
aCoder.encode(self.photo, forKey: PropertyKey.photo)
}
required convenience init?(coder aDecoder: NSCoder) {
// The name is required. If we cannot decode a name string, the initializer should fail.
guard let name = aDecoder.decodeObject(forKey: PropertyKey.name) as? String else {
os_log("unable to find name", log: OSLog.default, type: .debug)
return nil
}
guard let photo = aDecoder.decodeObject(forKey: PropertyKey.photo) as? UIImage else {
os_log("unable to find photo", log: OSLog.default, type: .debug)
return nil
}
// Must call designated initializer.
self.init(name: name, photo: photo)
}
init?(name: String, photo: UIImage) {
// Initialize stored properties.
self.name = name
self.photo = photo
}
}
您可以使用此方法取消归档数据。
https://developer.apple.com/documentation/foundation/nskeyedunarchiver/2963379-unarchivedobject
NSKeyedUnarchiver.unarchivedObject(ofClasses:from:)
您可以将解档代码写为;
let data = try Data(contentsOf: filePath)
let object = try NSKeyedUnarchiver.unarchivedObject(ofClasses:[object.self, UIImage.self], from: data)
希望它有效。
PS。关于命名 类 的一个提示。它们应始终以大写字母开头。
编辑:您还应该在尝试取消存档时添加 UIImage.self。
并且您应该如下所示更改存档方法。
NSKeyedArchiver.archivedData(withRootObject: data, requiringSecureCoding: true)
使用此解压方法时,您的模型还应满足 NSSecureCoding 协议。
extension object : NSSecureCoding {
static var supportsSecureCoding: Bool {
return true
}
}
在我的视图控制器中,我有一个数组中的自定义 class 存储。我使用 NSKeyedArchiver 来存储数据,但我使用的是折旧版本 NSKeyedUnarchiver.unarchiveObject(withFile: object.ArchiveURL.path) as? [object]
.
现在我决定更新为:
let data = try Data(contentsOf: filePath)
let object = try NSKeyedUnarchiver.unarchiveTopLevelObjectWithData(data)
这是我的保存功能:
let data = try NSKeyedArchiver.archivedData(withRootObject: object, requiringSecureCoding: false)
try data.write(to: filePath)
当我 return 从 Unarchiver 反对时,它 return 没有。我不确定为什么。 这是我的习惯 class:
class object: NSObject, NSCoding {
var name: String
var photo: UIImage
struct PropertyKey {
static let name = "name"
static let photo = "photo"
}
func encode(with aCoder: NSCoder) {
aCoder.encode(self.name, forKey: PropertyKey.name)
aCoder.encode(self.photo, forKey: PropertyKey.photo)
}
required convenience init?(coder aDecoder: NSCoder) {
// The name is required. If we cannot decode a name string, the initializer should fail.
guard let name = aDecoder.decodeObject(forKey: PropertyKey.name) as? String else {
os_log("unable to find name", log: OSLog.default, type: .debug)
return nil
}
guard let photo = aDecoder.decodeObject(forKey: PropertyKey.photo) as? UIImage else {
os_log("unable to find photo", log: OSLog.default, type: .debug)
return nil
}
// Must call designated initializer.
self.init(name: name, photo: photo)
}
init?(name: String, photo: UIImage) {
// Initialize stored properties.
self.name = name
self.photo = photo
}
}
您可以使用此方法取消归档数据。 https://developer.apple.com/documentation/foundation/nskeyedunarchiver/2963379-unarchivedobject
NSKeyedUnarchiver.unarchivedObject(ofClasses:from:)
您可以将解档代码写为;
let data = try Data(contentsOf: filePath)
let object = try NSKeyedUnarchiver.unarchivedObject(ofClasses:[object.self, UIImage.self], from: data)
希望它有效。
PS。关于命名 类 的一个提示。它们应始终以大写字母开头。
编辑:您还应该在尝试取消存档时添加 UIImage.self。 并且您应该如下所示更改存档方法。
NSKeyedArchiver.archivedData(withRootObject: data, requiringSecureCoding: true)
使用此解压方法时,您的模型还应满足 NSSecureCoding 协议。
extension object : NSSecureCoding {
static var supportsSecureCoding: Bool {
return true
}
}