NSKeyedUnarchiver данные в неправильном формате - PullRequest
1 голос
/ 19 сентября 2019

Я использую ARKit и GameKitMatches, поэтому я не могу использовать Codable (afaik), потому что MCPeerID и ARWorldMap не кодируются, чтобы сначала убрать это с пути.

Поэтому я использую NSCoding и NSSecureCoding, но по какой-то причине я всегда улавливаю ошибку:

The data couldn’t be read because it isn’t in the correct format.

... даже если я буквально только что создал ее.Я также пытался использовать NSKeyedUnarchiver.unarchivedObject(ofClasses: classes, но это вызвало неожиданный ноль в моем инициате.

Вот игровая площадка, которую я сделал, показывая проблему:

class CodingData: NSObject, NSCoding, NSSecureCoding {
    static var supportsSecureCoding = true
    var dic: [String: Int]!
    var i: Int!

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

    required convenience init?(coder: NSCoder) {
        let anInt = coder.decodeObject(forKey: "i") as! Int
        let anDic = coder.decodeObject(forKey: "dic") as! [String: Int]
        self.init(dic: anDic, i: anInt)
    }

    init(dic: [String: Int], i: Int){
        self.dic = dic
        self.i = i
    }
}

do{
    let test = CodingData(dic: [:], i: 0)
    //let classes = [NSDictionary.self, NSNumber.self]
    let testData = try NSKeyedArchiver.archivedData(withRootObject: test, requiringSecureCoding: true)
    let emptyDic = try NSKeyedUnarchiver.unarchivedObject(ofClass: CodingData.self, from: testData)
    // Error here ^^^^^^
}catch{
    error.localizedDescription
}

Кстати, не уверен, если это имеет значениено при попытке отладки coder в инициализации он всегда говорил (проб. просто ошибка):

error: <EXPR>:1:1: error: non-nominal type '$__lldb_context' (aka 'Self') cannot be extended
extension $__lldb_context {
^         ~~~~~~~~~~~~~~~

error: <EXPR>:19:27: error: value of type 'Self' has no member '$__lldb_wrapped_expr_28'
    $__lldb_injected_self.$__lldb_wrapped_expr_28(
    ~~~~~~~~~~~~~~~~~~~~~ ^~~~~~~~~~~~~~~~~~~~~~~

1 Ответ

1 голос
/ 20 сентября 2019

Вы приложили большие усилия, но в вашем коде так много неправильного, что я думаю, что самый простой способ ответить на него - просто исправить его:

class CodingData: NSObject, NSSecureCoding {
    static var supportsSecureCoding = true
    var dic: [String: Int]
    var i: Int

    func encode(with coder: NSCoder) {
        coder.encode(i as NSNumber, forKey: "i")
        coder.encode(dic as NSDictionary, forKey: "dic")
    }

    required init?(coder: NSCoder) {
        let anInt = coder.decodeObject(of: NSNumber.self, forKey: "i")
        let anDic = coder.decodeObject(of: NSDictionary.self, forKey: "dic")
        self.dic = anDic as! [String:Int]
        self.i = anInt as! Int
    }

    init(dic: [String: Int], i: Int){
        self.dic = dic
        self.i = i
    }
}

Это будет работать сейчас:

let test = CodingData(dic: [:], i: 0)
let testData = try NSKeyedArchiver.archivedData(withRootObject: test, requiringSecureCoding: true)
let emptyDic = try NSKeyedUnarchiver.unarchivedObject(ofClass: CodingData.self, from: testData)
print(emptyDic?.dic as Any, emptyDic?.i as Any) // Optional([:]) Optional(0)

Итак, вашими четырьмя основными ошибками были:

  • Не объявляйте свойство экземпляра как неявно развернутое Необязательное (!)

  • init?(coder:) не является инициализатором convenience

  • Для безопасного кодирования необходимо декодировать каждое закодированное свойство с использованием безопасного кодирования (decodeObject(of:forKey:))

  • Только типы NSSecureCoding могут быть закодированы с использованием безопасного кодирования, поэтому типы Swift должны быть приведены к их эквивалентам Objective C

...