webentwicklung-frage-antwort-db.com.de

Unarchive Array mit NSKeyedUnarchiver unarchivedObject (ofClass: from :)

Seit dem Upgrade auf Swift 4.2 habe ich festgestellt, dass viele der Methoden NSKeyedUnarchiver und NSKeyedArchiver nicht mehr empfohlen werden. Wir müssen nun die Typmethode static func unarchivedObject<DecodedObjectType>(ofClass: DecodedObjectType.Type, from: Data) -> DecodedObjectType? verwenden, um Daten aus dem Archiv zu entfernen.

Es ist mir gelungen, ein Array meiner benutzerdefinierten Klasse WidgetData, einer NSObject-Unterklasse, erfolgreich zu archivieren:

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

}

Das Problem tritt auf, wenn ich versuche, diese Daten aus dem Archiv zu entfernen:

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

}

Dies schlägt jedoch fehl, da die Verwendung von Array.self anstelle von NSArray.self nicht zulässig ist. Was mache ich falsch und wie kann ich das beheben, um mein Array aufzuheben?

4
Geoff H

Sie können unarchiveTopLevelObjectWithData(_:) verwenden, um die Archivierung der von archivedData(withRootObject:requiringSecureCoding:) archivierten Daten aufzuheben. (Ich glaube, das ist noch nicht veraltet.)

Bevor Sie jedoch etwas Code anzeigen, sollten Sie besser:

  • Vermeiden Sie die Verwendung von NSData, sondern verwenden Sie stattdessen Data

  • Vermeiden Sie die Verwendung von try?, der Fehlerinformationen enthält, die für das Debugging hilfreich sind

  • Entfernen Sie alle nicht mehr benötigten Abgüsse


Versuche dies:

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)")
    }
}

Wenn Sie jedoch eine neue App erstellen, sollten Sie Codable in Betracht ziehen.

6
OOPer
unarchiveTopLevelObjectWithData(_:) 

ist auch veraltet. Um Daten ohne sichere Codierung zu archivieren, müssen Sie:

  1. NSKeyedUnarchiver mit init(forReadingFrom: Data) erstellen
  2. Setzen Sie requiresSecureCoding des erstellten Unarchivers auf false.
  3. Rufen Sie decodeObject(of: [AnyClass]?, forKey: String) -> Any? auf, um Ihr Objekt abzurufen. Verwenden Sie einfach die richtige Klasse und NSKeyedArchiveRootObjectKeyas.
3
Maciej S
 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
        }

// Daten erhalten 

 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")
        }

Auf diese Weise habe ich meinen Code aktualisiert und funktioniert für mich

2
user1828845

Sie suchen wahrscheinlich danach:

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
        }
    }
1
Hopreeeenjust