Возникла проблема при разархивировании данных объекта - PullRequest
1 голос
/ 07 апреля 2020

Я создал generic function для NSKeyedArchiver и NSKeyedUnarchiver. Я могу заархивировать данные массива, но при выполнении разархивирования сталкиваюсь с проблемой. Ниже мой код:

Код NSKeyedArchiver:

func cacheData<T>(data: T) {        
        do {
            let codedData = try NSKeyedArchiver.archivedData(withRootObject: data, requiringSecureCoding: false)
        } catch {
            print("Exception while caching data \(error)")
        }
    }

Код NSKeyedUnarchiver:

func getCacheData<T>(encodedData: Data, ofClass: T.Type) -> [T]? {
    do{
        if let decodedData = try NSKeyedUnarchiver.unarchivedObject(ofClasses: [NSArray.self, T.self as! AnyClass], from: encodedData){
            return decodedData as? [T]
        }
    } catch {
        print("Exception while decode array cache data \(error)")
    }
    return nil
}

Над кодом работает хорошо для того, чтобы иметь только strings, integers переменные, но это не удалось, если иметь custom classes переменные. Как разрешить эти пользовательские классы в NSKeyedUnarchiver.

Я получаю ниже ошибку:

Исключение при декодировании данных кэша массива Ошибка Domain = NSCocoaErrorDomain Code = 4864 "значение для ключа" customclass1 'был неожиданного класса' CustomClass1 '. Допустимые классы:' {(NSArray, MainClass)} '. " UserInfo = {NSDebugDescription = значение для ключа 'customclass2' было неожиданного класса 'CustomClass2'. Допустимые классы: '{(NSArray, MainClass)}'.}

Есть идеи, как решить эту проблему?

1 Ответ

1 голос
/ 07 апреля 2020

Убедитесь, что все ваши классы подтверждают NSCoding. Как то так:

func archiveAndUnarchive() {
let class2 = Class2(value: "Value")
let class1 = Class1(name: "Name", class2: class2)
do {
    // ARCHIVING
    let data = try NSKeyedArchiver.archivedData(withRootObject: class1, requiringSecureCoding: false)

    // UNARCHIVING
    if let decodedData = try NSKeyedUnarchiver.unarchiveTopLevelObjectWithData(data) as? Class1 {
        print(decodedData)
    }
} catch {
    print(error)
}
}

class Class1: NSObject, NSCoding {
var name: String?
var class2: Class2?

func encode(with coder: NSCoder) {
    coder.encode(name, forKey: "name")
    coder.encode(class2, forKey: "class2")
}

required init?(coder: NSCoder) {
    super.init()
    self.name = coder.decodeObject(forKey: "name") as? String ?? ""
    self.class2 = coder.decodeObject(forKey: "class2") as? Class2
}

init(name: String, class2: Class2) {
    super.init()
    self.name = name
    self.class2 = class2
}
}


class Class2: NSObject, NSCoding {
var value: String?

func encode(with coder: NSCoder) {
    coder.encode(value, forKey: "value")
}

required init?(coder: NSCoder) {
    super.init()
    self.value = coder.decodeObject(forKey: "value") as? String
}

init(value: String) {
    super.init()
    self.value = value
}
}
...