线程安全队列

Thread-safe queues

我正在执行 20 个具有不同 URL 的数据任务,并用结果填充一个数组,但在函数结束时,尽管使用相同的队列,我仍然没有得到最终数组。

    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 行在数据任务的任何结果添加到队列之前立即执行。

您需要DispatchGroupnotify闭包在所有任务完成后执行。

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 个任务