前言
Swift 4是苹果计划于2017年秋季推出的最新版本,其主要重点是提供与Swift 3代码的源兼容性,并努力实现ABI稳定性。从Swift4开始提供的Decodable解析JSON确实很方便,但遇到一个小问题,记录一下。
当JSON中某个key的值为{}或者空字符串”“,而该值需要解析的不是基本类型时,即使标记为 Optional,依然会导致整个解析失败:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
|
//: 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而不是抛出错误导致整个解析失败:
1
2
3
4
5
|
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
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对服务器之家的支持。
原文链接:https://antscript.com/post/2018-03-14-iOS-decodable-empty-json/