同步执行GET请求

时间:2018-01-12 16:04:28

标签: ios swift synchronous

我有一个向API发送GET请求的函数,我需要等待响应才能继续使用其余的代码。

我尝试了各种不同的线程,但还没有让它发挥作用。我一直试图通过自己解决这个问题,但我对线程非常困惑,因此排在第四位。

这些是我的功能:

func checkDomains() {
    let endings = [".com"]
    for ending in endings {
        let domainName = names[randomNameIndx] + ending
        let urlString = "https://domainr.p.mashape.com/v2/status?mashape-key={my-key}" + domainName

        let url = URL(string: urlString)
        var request = URLRequest(url: url!)

        request.setValue("{my-key}", forHTTPHeaderField: "X-Mashape-Key")
        request.setValue("application/json", forHTTPHeaderField: "Accept")
        request.httpMethod = "GET"
        var ourBool = false

        DispatchQueue.main.sync {
            let task = URLSession.shared.dataTask(with: request, completionHandler: { (data, response, er) in

                do {
                    print("hey")
                    if let json = try JSONSerialization.jsonObject(with: data!, options: .allowFragments) as? NSDictionary {
                        ourBool = String(describing: json).contains("inactive")

                        if ourBool {
                            self.domainStatuses.append("available")
                        } else {
                            self.domainStatuses.append("taken")
                        }
                    }
                } catch {

                }


            })
            task.resume()
        }
    }
}

@objc func btnTapped(sender: UIButton) {
        self.prepareForNewName()
        sender.shake(direction: "horizontal", swings: 1)
        self.checkDomains() // This needs to finish before calling the next functions
        self.setupForNewName()
        self.animateForNewName()
}

3 个答案:

答案 0 :(得分:1)

我的建议是在你的异步函数中添加回调参数。示例如下:

func checkDomains(_ onResult: @escaping (Error?) -> Void)

然后,您可以在服务器返回成功结果或错误的位置调用函数内的onResult。

func checkDomains(_ onResult: @escaping (Error?) -> Void) {
    ...
    let task = URLSession.shared.dataTask ... {
       do {
           //Parse response
           DispatchQueue.main.async { onResult(nil) }
       } catch {
           DispatchQueue.main.async { onResult(error) }
       }
       ...
    }
}

你需要的最后一件事就是在调用checkDomains函数

的地方传递回调参数
@objc func btnTapped(sender: UIButton) {
    self.prepareForNewName()
    sender.shake(direction: "horizontal", swings: 1)
    self.checkDomains { [unowned self] error in
        if let error = error {
            // handle error
            return
        }
        self.setupForNewName()
        self.animateForNewName()
    }
}

答案 1 :(得分:0)

最近异步进行网络调用是一种好习惯。有一些技术可以使代码更简单,比如https://cocoapods.org/pods/ResultPromises TBC ...

答案 2 :(得分:0)

感谢您的回答。我刚刚回到家中,意识到有一种更简单的方法可以做到这一点。

let task = URLSession.shared.dataTask(with: request, completionHandler: { (data, response, error) in

    do {
            ...
            if response != nil {
                DispatchQueue.main.async {
                    self.setupForNewName()
                    self.animateForNewName()
                }
            }
        }
    } catch {
        print(error)
    }
})
task.resume()

当然这取决于API的响应,但我相信API总是会响应。