概述
前言
Swift 4是苹果计划于2017年秋季推出的最新版本,其主要重点是提供与Swift 3代码的源兼容性,并努力实现ABI稳定性。从Swift4开始提供的Decodable解析JSON确实很方便,但遇到一个小问题,记录一下。
当JSON中某个key的值为{}或者空字符串”“,而该值需要解析的不是基本类型时,即使标记为 Optional,依然会导致整个解析失败:
//: Playground import Foundation //Book.swift struct Book: Codable { var id: Int var name: String var cover: BookCover? enum CodingKeys: String, CodingKey { case id case name case cover } struct BookCover: Codable { var url: String var thumbURL: String enum CodingKeys: String, CodingKey { case url case thumbURL = "thumb_url" } } } //JSON let bookJSON1 = """ { "id": 1, "name": "fake name 1", "cover": { "url": "a.png", "thumb_url": "b.png" } } """ let bookJSON2 = """ { "id": 2, "name": "fake name 2", "cover": { } } """ //解析 let decoder = JSONDecoder() decoder.dateDecodingStrategy = .iso8601 let book1 = try? decoder.decode(Book.self, from: bookJSON1.data(using: .utf8)!) // 解析正常 print(book1) let book2 = try? decoder.decode(Book.self, from: bookJSON2.data(using: .utf8)!) // 输出 nil,cover已经是 Optional,为何整个book都解析失败? print(book2)
原因:
因为cover是 Optional,所以会调用 decodeIfPresent 来解析,而在cover节点中没有找到url,thumb_url这两个key,导致默认解析失败,直接抛出了错误。
解决:
重新实现 decodeIfPresent,在解析失败时返回nil而不是抛出错误导致整个解析失败:
extension KeyedDecodingContainer { public func decodeIfPresent<T>(_ type: T.Type, forKey key: K) throws -> T? where T : Decodable { return try? decode(type, forKey: key) } }
参考: http://davelyon.net/2017/08/16/jsondecoder-in-the-real-world
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对靠谱客的支持。
最后
以上就是舒服树叶为你收集整理的Swift利用Decodable解析JSON的一个小问题详解的全部内容,希望文章能够帮你解决Swift利用Decodable解析JSON的一个小问题详解所遇到的程序开发问题。
如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。
本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
发表评论 取消回复