处理多个完成处理程序

时间:2017-11-20 02:14:02

标签: swift concurrency completionhandler dispatch-queue

我正在尝试为数组中的每个元素协调几个完成处理程序。

代码基本上是这样的:

.len()

因此,Array中的每个元素都通过带有完成处理程序的服务发送,并且所有结果都存储在一个数组中。完成所有这些处理程序后,我希望执行一些代码。

我尝试使用var results = [String:Int]() func requestData(for identifiers: [String]) { identifiers.forEach { identifier in service.request(identifier, completion: { (result) in result[identifier] = result }) } // Execute after all the completion handlers finish print(result) }

执行此操作
DispatchQueue

但打印调用仍在执行,空var results = [String:Int]() func requestData(for identifiers: [String]) { let queue = DispatchQueue.init(label: "queue") identifiers.forEach { identifier in service.request(identifier, completion: { (result) in queue.sync { result[identifier] = result } }) } // Execute after all the completion handlers finish queue.sync { print(result) } }

2 个答案:

答案 0 :(得分:7)

如果我了解你正在尝试做什么,你可能想要使用DispatchGroup

以下是一个例子:

let group = DispatchGroup()

var letters = ["a", "b", "c"]

for letter in letters {
    group.enter()
    Server.doSomething(completion: { [weak self] (result) in
        print("Letter is: \(letter)")
        group.leave()
    })
}

group.notify(queue: .main) {
    print("- done")
}

这将打印如下内容:

b
c
a
// ^ in some order
- done

答案 1 :(得分:0)

首先,请注意您的service.request(...)是以异步模式处理的。另一个问题是您希望在该循环中完成所有服务请求。

我的建议是使用完成处理程序创建函数,并在每个完成的循环上添加一个计数器。您的功能与以下类似。

var results = [String:Int]()

func requestData(for identifiers: [String], callback:@escaping (Bool) -> Void)
{
    var counter = 0
    var maxItem = identifiers.count

    identifiers.forEach
    {   identifier in

        service.request(identifier, completion: { (result) in
            result[identifier] = result
            counter += 1
            if counter == maxItem {
                callback(true) // update completion handler to say all loops request are done
            }
            // if not, continue the other request
        })
    }
}

这是代码的另一部分将如何调用该函数并等待回调

requestData(for identifiers:yourArrays) { (complete) in

    if complete {
        print(results)
    }
}

如果发生错误,请不要忘记管理。