我正在用不同的url执行20个数据任务,并用结果填充一个数组,但是在函数的末尾,尽管使用了相同的队列,我仍然没有得到最终的数组。
class func getImages(photos:[Photo]){
var images:[UIImage?] = []
let arrayQueue = DispatchQueue(label: "imagesQueue",attributes: .concurrent)
for i in 0...20{
let url = EndPoints.getImage(farm: photos[i].farm, server: photos[i].server, id: photos[i].id, secret: photos[i].secret).url
let task = URLSession.shared.dataTask(with: url) { (data, response, error) in
arrayQueue.async(flags: .barrier) {
let image = UIImage(data: data!)!
images.append(image)
}
}
task.resume()
}
arrayQueue.sync {
print(images.count)
}
}
这个函数应该打印
21
,但是它打印零。 最佳答案
当然是因为函数的结尾不是时间轴的结尾。方法末尾的print
行在数据任务的任何结果添加到队列之前立即执行。
您需要DispatchGroup
,在所有任务完成后执行notify
闭包。
class func getImages(photos:[Photo]) {
var images:[UIImage] = [] // Why optional?
let group = DispatchGroup()
let arrayQueue = DispatchQueue(label: "imagesQueue", attributes: .concurrent)
for i in 0...20{
let url = EndPoints.getImage(farm: photos[i].farm, server: photos[i].server, id: photos[i].id, secret: photos[i].secret).url
group.enter()
let task = URLSession.shared.dataTask(with: url) { (data, response, error) in
arrayQueue.async(flags: .barrier) {
let image = UIImage(data: data!)!
images.append(image)
group.leave()
}
}
task.resume()
}
group.notify(queue: arrayQueue) {
print(images.count)
}
}
请注意,当发生错误时,您的代码会可靠地崩溃。。。
而且您的自定义队列实际上是多余的,您正在执行21个任务
关于ios - 线程安全队列,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/56199349/