我正在用不同的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/

10-10 20:43
查看更多