我正在解析表示事件的json数据的提要。在发生事件的一天,json提要将是一个Dictionary of Dictionary,看起来像这样:
{
"19374176" :
{
"event_title" : "Cool Fun Thing to Do",
"event_description" : "Have fun and do something cool",
"event_start_time" : "13:00:00",
"event_end_time" : "14:00:00"
},
"90485761634" :
{
"event_title" : "Nap Time",
"event_description" : "Lay down and go to sleep.",
"event_start_time" : "15:00:00",
"event_end_time" : "16:00:00"
}
}
我已经建立了结构,可以使用当前作为较大功能一部分的代码的方式解码和使用此信息:
URLSession.shared.dataTask(with: url){(data, response, error) in
if error != nil {
print("session error: ", error!.localizedDescription)
}
guard let data = data else { return }
do {
let decoder = JSONDecoder()
decoder.keyDecodingStrategy = .convertFromSnakeCase
print(data)
var eventData = try decoder.decode([String:Event].self, from: data)
DispatchQueue.main.async{
self.events = Array(eventData.values).sorted(by: {$0.timeStart < $1.timeStart
})
self.updateView()
self.refreshControl.endRefreshing()
self.activityIndicatorView.stopAnimating()
}
} catch DecodingError.typeMismatch(let type, let context){
//No Dictionary of Events in Data
print("key:", type, "context: ", context)
} catch let jsonError{
print("json error: ", jsonError)
}
}.resume()
}
我现在的问题是,在没有事件的日子里,json提要是一个空数组:
[]
这会导致我处理的类型不匹配,但是如果我尝试从catch中调用updateView,refreshControl或activityIndicatorView的函数,则会收到一个错误,我无法在主线程之外调用它们。
我尝试做嵌套的try块来分配eventData变量(首先查看它是否为[String]((空数组,没有事件)),然后再分配给[String:Any]((具有事件值的数组))),但这给了URLSession一个错误。
是否有更好的方法来查看json是否为空数组而不是事件值的填充字典并无论如何更新视图?
答案 0 :(得分:3)
尝试在任何块之外刷新它们
do {
let decoder = JSONDecoder()
decoder.keyDecodingStrategy = .convertFromSnakeCase
print(data)
var eventData = try decoder.decode([String:Event].self, from: data)
self.events = Array(eventData.values).sorted(by: {$0.timeStart < $1.timeStart
} catch DecodingError.typeMismatch(let type, let context){
//No Dictionary of Events in Data
print("key:", type, "context: ", context)
} catch let jsonError{
print("json error: ", jsonError)
}
DispatchQueue.main.async{
self.updateView()
self.refreshControl.endRefreshing()
self.activityIndicatorView.stopAnimating()
}