当我有一个异步请求时,我不能更改TabBarViewController

时间:2019-01-28 16:11:18

标签: swift grand-central-dispatch codable

在我提出请求并找到并填充对象之前,我无法更改视图。

我试图将代码与GCD异步。那没用

override func viewDidLoad() {
    getHeartStroke()
NotificationCenter.default.addObserver(forName:NSNotification.Name("HeartStrokeNotification"), object: nil, queue: nil, using: notificationFinish)
 }

 func getHeartStroke() {
AF.request("http://localhost:8080/heartstroke", method: .get, headers: nil).responseJSON(completionHandler: {response in
    if (response.error == nil)
    {
        let json = JSON(response.result.value!)
            DispatchQueue.global(qos: .userInitiated).async {
                guard let hearstrokeArray = try? JSONDecoder().decode([HeartStroke].self, from: json.rawData()) else{
                    debugPrint("An error has occurred")
                    return
                }
                NotificationCenter.default.post(name:NSNotification.Name("HeartStrokeNotification"), object: hearstrokeArray, userInfo: nil)
            }
        }else{
            NotificationCenter.default.post(name:NSNotification.Name("HeartStrokeErrorNotification"), object: nil, userInfo: nil)
        }
    })
 }

 func notificationFinish(notification:Notification) -> Void{
  if (notification.name.rawValue == "HeartStrokeNotification"){

    arrayHeartstroke = notification.object as! [HeartStroke]
    DispatchQueue.main.async(execute: {
        self.tableView.reloadData()
    })
}

有了这段代码,我一直停留在页面上,直到getHeartStroke()结束为止,我希望在提取的同时浏览我的应用程序。

1 个答案:

答案 0 :(得分:0)

您需要一个完成处理程序来处理此问题。使用通知中心只会使您的生活变得困难和复杂,并可能导致意外行为。这是一些示例代码:

func getHeartStroke(completionHandler: (_ heartStroke: [HeartStroke?], _ error: NSError?) -> ()) {
    AF.request("http://localhost:8080/heartstroke", method: .get, headers: nil).responseJSON(completionHandler: {response in
        if (response.error == nil)
        {
            let json = JSON(response.result.value!)
            DispatchQueue.global(qos: .userInitiated).async {
                guard let hearstrokeArray = try? JSONDecoder().decode([HeartStroke].self, from: json.rawData()) else{
                    debugPrint("An error has occurred")
                    return
                }
                completionHandler(hearstrokeArray, nil)
            }
        } else {
            completionHandler(nil, response.error))
        }
    })
}

然后您可以这样称呼它:

getHeartStroke { [weak self] (heartStrokeArray, error) in
     guard let self = self else {return}
        if error != nil {
            self.processError(error)
        } else {
            self.processHeartStroke(heartStrokeArray)
        }
    }

processError和processHeartStroke将是您应创建的用于处理heartStrokeArray和错误对象的函数。

这些是标准回调或将函数传递到函数中。您在网上发现的许多课程似乎都忽略了回调,但是绝对值得您花些时间来学习它们。

您可以在此处了解有关闭包的更多信息(此处为completionHandler,在这里名为:completionHandler)