Ошибка typeMismatch при анализе данных JSON в Swift 4.2 - PullRequest
0 голосов
/ 30 апреля 2019

Ошибка несоответствия типов - ожидается декодирование массива, но найден словарь. Я использую локальный файл JSON, встроенный в код. Файл Json содержит информацию о пакетах. Мне нужно выбрать значения и показать их соответственно в контроллере таблицы. Пожалуйста, укажите, что я делаю неправильно в кодируемой модели или в коде. Разбор JSON не обработан должным образом.

Формат файла:

{
  "packages": [
    {
      "name": "Platinum Maksi 6 GB",
      "desc": "Zengin içerikli Platinum Maksi Paketi ile Turkcell Uygulamalarının keyfini sürün!",
      "subscriptionType": "monthly",
      "didUseBefore": true,
      "benefits": [
        "TV+",
        "Fizy",
        "BiP",
        "lifebox",
        "Platinum",
        "Dergilik"
      ],
      "price": 109.90,
      "tariff": {
        "data": "6144",
        "talk": "2000",
        "sms": "100"
      },
      "availableUntil": "1558131150"
    }
]
}

Модель: Базовая модель

struct Base : Codable {

    let packages : [Package]?

    enum CodingKeys: String, CodingKey {
            case packages = "packages"
    }

    init(from decoder: Decoder) throws {
            let values = try decoder.container(keyedBy: CodingKeys.self)
            packages = try values.decodeIfPresent([Package].self, forKey: .packages)
    }

2) Модель упаковки:

struct Package : Codable {

    let availableUntil : String?
    let benefits : String?
    let desc : String?
    let didUseBefore : Bool?
    let name : String?
    let price : Int?
    let subscriptionType : String?
    let tariff : Tariff?

    enum CodingKeys: String, CodingKey {
            case availableUntil = "availableUntil"
            case benefits = "benefits"
            case desc = "desc"
            case didUseBefore = "didUseBefore"
            case name = "name"
            case price = "price"
            case subscriptionType = "subscriptionType"
            case tariff = "tariff"
    }

    init(from decoder: Decoder) throws {
            let values = try decoder.container(keyedBy: CodingKeys.self)
            availableUntil = try values.decodeIfPresent(String.self, forKey: .availableUntil)
            benefits = try values.decodeIfPresent(String.self, forKey: .benefits)
            desc = try values.decodeIfPresent(String.self, forKey: .desc)
            didUseBefore = try values.decodeIfPresent(Bool.self, forKey: .didUseBefore)
            name = try values.decodeIfPresent(String.self, forKey: .name)
            price = try values.decodeIfPresent(Int.self, forKey: .price)
            subscriptionType = try values.decodeIfPresent(String.self, forKey: .subscriptionType)
            tariff = try Tariff(from: decoder)
    }

}

Тарифная модель:

struct Tariff : Codable {

    let data : String?
    let sms : String?
    let talk : String?

    enum CodingKeys: String, CodingKey {
            case data = "data"
            case sms = "sms"
            case talk = "talk"
    }

    init(from decoder: Decoder) throws {
            let values = try decoder.container(keyedBy: CodingKeys.self)
            data = try values.decodeIfPresent(String.self, forKey: .data)
            sms = try values.decodeIfPresent(String.self, forKey: .sms)
            talk = try values.decodeIfPresent(String.self, forKey: .talk)
    }

}

Мой код:

var pack = [Package]()

override func viewDidLoad() {
    super.viewDidLoad()
    // Do any additional setup after loading the view.

    let url = Bundle.main.url(forResource: "packageList", withExtension: "json")!
    do {
        let data = try Data(contentsOf: url)
        pack = try JSONDecoder().decode([Package].self, from: data)
        print(pack)

    } catch {
        print(error)
    }

1 Ответ

0 голосов
/ 30 апреля 2019

Вы декодируете не тот объект.Вы должны всегда декодировать корневой объект JSON, который является Base.

Получить массив Package с помощью base.packages

let base = try JSONDecoder().decode(Base.self, from: data)
pack = base.packages

Теперь вы получите два других несоответствия типовошибки, измените тип benefits на [String] и price на Double.

Вы можете значительно сократить свои структуры: удалите все CodingKeys и все инициализаторы и объявите все элементы структуры как не-опциональный (убрать вопросительные знаки).

Редактировать:

Чтобы декодировать availableUntil как Date, объявить его как Date

let availableUntil: Date

и добавьте пользовательскую стратегию декодирования даты , поскольку значение является строкой (было бы проще, если бы значение было Int)

let decoder = JSONDecoder()
decoder.dateDecodingStrategy = .custom{ decoder -> Date in
    let container = try decoder.singleValueContainer()
    let dateStr = try container.decode(String.self)
    guard let interval = TimeInterval(dateStr) else { throw DecodingError.dataCorruptedError(in: container, debugDescription: "Date string cannot be converted to TimeInterval") }
    return Date(timeIntervalSince1970: interval)
}
let data = try Data(contentsOf: url)
let base = try decoder.decode(Base.self, from: data)
pack = base.packages
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...