如何将JSONDecoder与多种可能的数据结构一起使用

时间:2019-05-16 14:18:05

标签: json swift decodable

我正在解析表示事件的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或activityIndi​​catorView的函数,则会收到一个错误,我无法在主线程之外调用它们。

我尝试做嵌套的try块来分配eventData变量(首先查看它是否为[String]((空数组,没有事件)),然后再分配给[String:Any]((具有事件值的数组))),但这给了URLSession一个错误。

是否有更好的方法来查看json是否为空数组而不是事件值的填充字典并无论如何更新视图?

1 个答案:

答案 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()
}