swift - 同步异步函数

标签 swift swift3 grand-central-dispatch

我尝试用 swift 3 创建我的第一个项目。

我尝试从我的 API 获取数据。如果我手动启动该功能,效果会很好。我需要同步异步请求。

我需要触发我的功能 3 次并等待其他人完成。

makeGetCall(URLstring: "api1")

等待完成

makeGetCall(URLstring: "api2")

等待完成

makeGetCall(URLstring: "api3") 

将其设置为背景并每 5 秒触发一次。

func makeGetCall(URLstring: String, update: Bool) {

    let completeURL = "http://myapi/" + URLstring


    // Set up the URL request
    guard let url = URL(string: completeURL) else {
        print("Error: cannot create URL")
        return
    }
    let urlRequest = URLRequest(url: url)

    // set up the session
    let config = URLSessionConfiguration.default
    let session = URLSession(configuration: config)

    // make the request
    let task = session.dataTask(with: urlRequest) {
        (data, response, error) in
        // check for any errors
        guard error == nil else {
            print("error calling GET on /todos/1")
            print(error as Any)
            return
        }
        // make sure we got data
        guard let responseData = data else {
            print("Error: did not receive data")
            return
        }

            // parse the result as XML
        if URLstring == "devicelist.cgi" {
            self.readDevice(XMLData: responseData)
        }

        if URLstring == "statelist.cgi" {
            self.readDeviceData(XMLData: responseData, update: update)
        }

        if URLstring == "functionlist.cgi" {
            self.readGewerke(XMLData: responseData)
        }
    }

    task.resume()

}

有人可以帮忙吗

哈根

这是我对完成处理程序的尝试:

    override func viewDidLoad() {
    super.viewDidLoad()

    makeGetCall(input: "statelist.cgi") {
        (result: Bool) in
        print("finished statelist")

    }
    makeGetCall(input: "devicelist.cgi") {
        (result: Bool) in
        print("finished devicelist")
    }


    makeGetCall(input: "functionlist.cgi") {
        (result: Bool) in
        print("finished functionlist")
    }

}

func makeGetCall(input: String, completion: @escaping (_ result: Bool) -> Void) {



    let completeURL = "http://192.168.0.25/addons/xmlapi/" + input


    // Set up the URL request
    guard let url = URL(string: completeURL) else {
        print("Error: cannot create URL")
        return
    }
    let urlRequest = URLRequest(url: url)

    // set up the session
    let config = URLSessionConfiguration.default
    let session = URLSession(configuration: config)

    // make the request
    let task = session.dataTask(with: urlRequest) {
        (data, response, error) in
        // check for any errors
        guard error == nil else {
            print("error calling GET on /todos/1")
            print(error as Any)
            return
        }
        // make sure we got data
        guard data != nil else {
            print("Error: did not receive data")
            return
        }
         completion(true)

    }

    task.resume()


}

如果我将这 3 个调用放在一起,它就会正常工作。 我认为它也应该适用于 GCD,但大多数示例适用于 swift 2。

    makeGetCall(input: "devicelist.cgi") {
        (result: Bool) in
        print("finished devicelist")
        self.makeGetCall(input: "functionlist.cgi") {
            (result: Bool) in
            print("finished functionlist")
           self.makeGetCall(input: "statelist.cgi") {
                (result: Bool) in
                print("finished statelist")

            }
        }
    }

也许现在有人可以提供帮助。

谢谢哈根

最佳答案

使用 DispatchGroup 在发生某些事情时得到通知。

override func viewDidLoad() {
    super.viewDidLoad()

    let stateListGroup = DispatchGroup()

    stateListGroup.enter()
    makeGetCall(input: "statelist.cgi") {
        (result: Bool) in
        print("finished statelist")
        stateListGroup.leave()
    }

    let deviceListGroup = DispatchGroup()
    deviceListGroup.enter()

    // the notify closure is called when the (stateList-) groups enter and leave counts are balanced.
    stateListGroup.notify(queue: DispatchQueue.main)  {
        self.makeGetCall(input: "devicelist.cgi") {
            (result: Bool) in
            print("finished devicelist")
            deviceListGroup.leave()
        }
    }

    let functionListGroup = DispatchGroup()
    functionListGroup.enter()

    deviceListGroup.notify(queue: DispatchQueue.main)  {
        self.makeGetCall(input: "functionList") {
            (result: Bool) in
            print("finished functionlist")
            functionListGroup.leave()
        }
    }

    functionListGroup.notify(queue: DispatchQueue.main)  {
        print("update ui here")
    }
}

打印:

statelist.cgi
finished statelist
devicelist.cgi
finished devicelist
functionList
finished functionlist
update ui here

另外请记住,session.dataTask() 的完成处理程序是在后台队列上调用的,因此我建议在主线程上调度 completion(true)排队以避免意外行为:

DispatchQueue.main.async {
    completion(true)
}

关于swift - 同步异步函数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42146274/

相关文章:

ios - AVCaptureSession 和相机线程未关闭 [iOS]

ios - 如何在 Swift 2.0 中创建一个 dispatch_block_t 数组?

ios - 如何将具有多个属性的枚举中的图像传递给 UIImageView

ios - 此类不符合 key 表 Swift 3 的键值编码

ios - 自定义 UITableViewCell - 不显示

ios - Collection View 。 Item 之间的最大距离。 swift 3

ios - 如何自动调整单元格大小以适合每行 4 个单元格

主队列上的 iOS 线程冲突

iOS 11 - 使用大标题作为导航 (Swift)

ios - 如何使用RxSwift取消之前的请求