Как использовать Swift Codable с вложенной структурой JSON и неизвестными ключами - PullRequest
0 голосов
/ 09 мая 2019

У меня есть вложенная структура JSON.Я знаю ключи верхнего уровня (объекты), но они могут присутствовать или не присутствовать для каждой выборки.Каждый ключ внутри объектов (и те, которые вложены в них) неизвестен.Они динамичны.

Я часами пытался использовать Codable только для того, чтобы вместо этого использовать JSONSerialization.Прежде чем я полностью потерял надежду, я хотел посмотреть, есть ли у кого-нибудь решение для этого.

Вот пример того, как выглядит мой JSON:

var jsonString =
"""
{
    "someNumbers": {
        "22": 6,
        "22626": 0
    },
    "someNestedAny": {
        "61": {
            "browser": 2
        },
        "8310": {
            "desktop": 2
        }
    },
    "someNestedArray": {
        "49": ["Chrome"],
        "50": ["Mac OS X"],
        "51": ["Mac desktop"],
        "52": ["browser"],
        "53": ["Chrome"]
    }
}
""" 

Пары значений ключа в каждом объекте верхнего уровня (someNumbers, someNestedAny и someNestedArray являются динамическими. Ключи / значения в пределахэти объекты также являются динамическими ... и т. д. Каждый объект верхнего уровня также является необязательным.

Я пробовал много вещей, но это то, что выглядело наиболее многообещающе (ни один из них не работал, хотя)

  1. 
    struct TopLevel: Decodable {
        var someNumbers: SomeNumbers?
        var someNestedAny: SomeNestedAny?
        var someNestedArray: SomeObjectFromNestedAny?
    }
    
    struct SomeNumbers: Decodable {
        var key: String
        var value: Int
    }
    
    struct SomeNestedAny: Decodable {
        var key: String
        var value: SomeObjectFromNestedAny
    }
    
    struct SomeObjectFromNestedAny: Decodable {
        var key: String
        var value: Int
    }
    
    struct SomeNestedArray: Decodable {
        var key: String
        var value: [String]
    }
    
    let data = jsonString.data(using: .utf8)!
    do {
        let result = try JSONDecoder().decode(TopLevel.self, from: data)
        print(result)
    } catch {
        print(error)
    }
    

Выход: keyNotFound(CodingKeys(stringValue: "key", intValue: nil), Swift.DecodingError.Context(codingPath: [CodingKeys(stringValue: "someNumbers", intValue: nil)], debugDescription: "No value associated with key CodingKeys(stringValue: \"key\", intValue: nil) (\"key\").", underlyingError: nil))

struct TopLevel: Decodable {
    var someNumbers: SomeNumbers?
    var someNestedAny: SomeNestedAny?
    var someNestedArray: SomeObjectFromNestedAny?
}

struct SomeNumbers: Decodable {

    public var numbersObject: [String: NumberKeys]

    public struct NumberKeys: Decodable {
        public let key: String
        public let value: Int
    }

    private struct NumberCodingKeys: CodingKey {
        var stringValue: String

        init?(stringValue: String) {
            self.stringValue = stringValue
        }

        var intValue: Int?

        init?(intValue: Int) {
            return nil
        }
    }

    public init(from decoder: Decoder) throws {
        let container = try decoder.container(keyedBy: NumberCodingKeys.self)
        self.numbersObject = [String: NumberKeys]()
        for key in container.allKeys {
            let value = try container.decode(NumberKeys.self, forKey: NumberCodingKeys(stringValue: key.stringValue)!)
            self.numbersObject[key.stringValue] = value
        }

    }
}

struct SomeNestedAny: Decodable {
    // was going to do the same thing as above - but it didn't work
    var key: String
    var value: SomeObjectFromNestedAny
}

struct SomeObjectFromNestedAny: Decodable {
    // was going to do the same thing as above - but it didn't work
    var key: String
    var value: Int
}

struct SomeNestedArray: Decodable {
    // was going to do the same thing as above - but it didn't work
    var key: String
    var value: [String]
}

let data = jsonString.data(using: .utf8)!
do {
    let result = try JSONDecoder().decode(TopLevel.self, from: data)
    print(result)
} catch {
    print(error)
}

Вывод: typeMismatch(Swift.Dictionary<Swift.String, Any>, Swift.DecodingError.Context(codingPath: [CodingKeys(stringValue: "someNumbers", intValue: nil), NumberCodingKeys(stringValue: "22626", intValue: nil)], debugDescription: "Expected to decode Dictionary<String, Any> but found a number instead.", underlyingError: nil))

Вот что у меня сейчас работает, но на самом деле брутто:

let data = jsonString.data(using: .utf8)
let json = try JSONSerialization.jsonObject(with: data!) as! [String:Any]
if let numbers = json["someNumbers"] as? [String:Any] {
    for (key, value) in numbers {
        print("key: \(key), value: \(value)")
    }
}
if let anys = json["someNestedAny"] as? [String: Any] {
    //print("tkey: \(anys)")
    for (key, value) in anys {
        //print("tkey2: \(key), tvalue: \(value)")
        if let value = value as? [String: Any] {
            let anyData = try JSONSerialization.data(withJSONObject: value, options: .prettyPrinted)
            let anyJson = try JSONSerialization.jsonObject(with: anyData) as! [String: Any]
            for (key2, value2) in anyJson {
                print("key: \(key), object-key: \(key2), object-value: \(value2)")
            }
        }
    }
}
if let arrays = json["someNestedArray"] as? [String: Any] {
    for (key, value) in arrays {
        print("key: \(key), value: \(value)")
    }
}

Спасибо зазаранее!

1 Ответ

0 голосов
/ 11 мая 2019

Трудно сказать, решает ли это ваш конкретный вариант использования, поскольку я не уверен, какова цель данных и как вы ожидаете их использовать, но оба приведенных ниже решения правильно декодируют ваш JSON в болеепригодный для использования объект Swift.

Самый простой способ - смоделировать его в точности как представленную вами структуру данных.Например, похоже, что someNumbers - это необязательный словарь с ключом String, со значениями Int: [String: Int]?.

struct TopLevel: Decodable {
  var someNumbers: [String: Int]?
  var someNestedAny: [String: [String: Int]]?
  var someNestedArray: [String: [String]]?
}

. Для большей читабельности при передаче объектов вокруг, вы можете добавитьнекоторые псевдонимы типов, и он становится

typealias SomeNumbers = [String: Int]
typealias SomeNestedAny = [String: [String: Int]]
typealias SomeNestedArray = [String: [String]]

struct TopLevel: Decodable {
  var someNumbers: SomeNumbers?
  var someNestedAny: SomeNestedAny?
  var someNestedArray: SomeNestedArray?
}

Чтобы получить полезные вещи, вам нужно будет вызвать такие вещи, как

topLevel.someNumbers?["22"]                 // 6
topLevel.someNestedAny?["8310"]             // ["desktop": 2]
topLevel.someNestedAny?["8310"]?["desktop"] // 2
topLevel.someNestedArray?["52"]             // ["browser"]
topLevel.someNestedArray?["52"]?[0]         // "browser"

Или, в зависимости от ваших потребностей, может иметь больше смысла в циклесквозь вещи

topLevel.someNestedAny?
  .forEach { item in
    print("|- \(item.key)")
    item.value.forEach { any in
      print("|  |- \(any.key)")
      print("|  |  |- \(any.value)")
    }
}

// |- 61
// |  |- browser
// |  |  |- 2
// |- 8310
// |  |- desktop
// |  |  |- 2

...