我正在尝试为数组中的每个元素协调几个完成处理程序。
代码基本上是这样的:
.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)
}
}
答案 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)
}
}
如果发生错误,请不要忘记管理。