Depuis la mise à niveau vers Swift 4.2, j'ai constaté que de nombreuses méthodes NSKeyedUnarchiver et NSKeyedArchiver étaient obsolètes et nous devons maintenant utiliser la méthode type static func unarchivedObject<DecodedObjectType>(ofClass: DecodedObjectType.Type, from: Data) -> DecodedObjectType?
pour désarchiver les données.
J'ai réussi à archiver avec succès un tableau de ma classe WidgetData, qui est une sous-classe NSObject:
private static func archiveWidgetDataArray(widgetDataArray : [WidgetData]) -> NSData {
guard let data = try? NSKeyedArchiver.archivedData(withRootObject: widgetDataArray as Array, requiringSecureCoding: false) as NSData
else { fatalError("Can't encode data") }
return data
}
Le problème survient lorsque j'essaie de désarchiver ces données:
static func loadWidgetDataArray() -> [WidgetData]? {
if isKeyPresentInUserDefaults(key: USER_DEFAULTS_KEY_WIDGET_DATA) {
if let unarchivedObject = UserDefaults.standard.object(forKey: USER_DEFAULTS_KEY_WIDGET_DATA) as? Data {
//THIS FUNCTION HAS NOW BEEN DEPRECATED:
//return NSKeyedUnarchiver.unarchiveObject(with: unarchivedObject as Data) as? [WidgetData]
guard let nsArray = try? NSKeyedUnarchiver.unarchivedObject(ofClass: NSArray.self, from: unarchivedObject as Data) else {
fatalError("loadWidgetDataArray - Can't encode data")
}
guard let array = nsArray as? Array<WidgetData> else {
fatalError("loadWidgetDataArray - Can't get Array")
}
return array
}
}
return nil
}
Mais cela échoue car l'utilisation de Array.self
au lieu de NSArray.self
est interdite. Qu'est-ce que je fais de mal et comment puis-je résoudre ce problème pour désarchiver mon tableau?
Vous pouvez utiliser unarchiveTopLevelObjectWithData(_:)
pour désarchiver les données archivées par archivedData(withRootObject:requiringSecureCoding:)
. (Je crois que ce n'est pas encore obsolète.)
Mais avant de montrer du code, vous devriez mieux:
Évitez d'utiliser NSData
, utilisez plutôt Data
Évitez d'utiliser try?
qui dispose d'informations d'erreur utiles pour le débogage
Supprimer tous les moulages inutiles
Essaye ça:
private static func archiveWidgetDataArray(widgetDataArray : [WidgetData]) -> Data {
do {
let data = try NSKeyedArchiver.archivedData(withRootObject: widgetDataArray, requiringSecureCoding: false)
return data
} catch {
fatalError("Can't encode data: \(error)")
}
}
static func loadWidgetDataArray() -> [WidgetData]? {
guard
isKeyPresentInUserDefaults(key: USER_DEFAULTS_KEY_WIDGET_DATA), //<- Do you really need this line?
let unarchivedObject = UserDefaults.standard.data(forKey: USER_DEFAULTS_KEY_WIDGET_DATA)
else {
return nil
}
do {
guard let array = try NSKeyedUnarchiver.unarchiveTopLevelObjectWithData(unarchivedObject) as? [WidgetData] else {
fatalError("loadWidgetDataArray - Can't get Array")
}
return array
} catch {
fatalError("loadWidgetDataArray - Can't encode data: \(error)")
}
}
Mais si vous créez une nouvelle application, vous devriez envisager d'utiliser Codable
.
unarchiveTopLevelObjectWithData(_:)
est obsolète aussi. Donc, pour désarchiver des données sans codage sécurisé, vous devez:
NSKeyedUnarchiver
avec init(forReadingFrom: Data)
requiresSecureCoding
of unarchiver créé sur false.decodeObject(of: [AnyClass]?, forKey: String) -> Any?
pour obtenir votre objet, utilisez simplement la classe et la clé NSKeyedArchiveRootObjectKey
as. if #available(iOS 12.0, *) {
guard let unarchivedFavorites = try? NSKeyedUnarchiver.unarchiveTopLevelObjectWithData(favoritesData!)
else {
return
}
self.channelFavorites = unarchivedFavorites as! [ChannelFavorite]
} else {
if let unarchivedFavorites = NSKeyedUnarchiver.unarchiveObject(with: favoritesData!) as? [ChannelFavorite] {
self.channelFavorites = unarchivedFavorites
}
// Atteindre des données
if #available(iOS 12.0, *) {
// use iOS 12-only feature
do {
let data = try NSKeyedArchiver.archivedData(withRootObject: channelFavorites, requiringSecureCoding: false)
UserDefaults.standard.set(data, forKey: "channelFavorites")
} catch {
return
}
} else {
// handle older versions
let data = NSKeyedArchiver.archivedData(withRootObject: channelFavorites)
UserDefaults.standard.set(data, forKey: "channelFavorites")
}
C’est ainsi que j’ai mis à jour mon code et que cela fonctionne pour moi
Vous recherchez probablement ceci:
if let widgetsData = UserDefaults.standard.data(forKey: USER_DEFAULTS_KEY_WIDGET_DATA) {
if let widgets = (try? NSKeyedUnarchiver.unarchivedObject(ofClasses: [NSArray.self, WidgetData.self], from: widgetsData)) as? [WidgetData] {
// your code
}
}