从异步调用修改数组,而不修改数组swift

时间:2019-05-29 23:02:43

标签: arrays swift asynchronous grand-central-dispatch

我正在尝试使用另一个异步调用的结果修改从异步调用检索的对象数组。基本上,我检索结果数组,并且有一个域messages从服务器调用中返回nil。然后,我需要在for循环中使用每个result id 进行另一个服务器调用,以获取messages数组。然后,将messages设置为返回值。

大约20分钟前,我刚刚从https://stackoverflow.com/a/48718976/3272438了解到使用 DispatchGroup()的习惯,所以请耐心等待。

我遇到的问题是,当我执行 print(“ C0:(self.res)”)时,它会打印出所有项目,包括我刚刚添加的messages。在group.notify(...)中打印 print(“ C2:(self.res)”)时,所有messages字段均显示为 nil 。我很沮丧,并且尝试了所有我能想到的。任何帮助将不胜感激。

更新 通过更改以下group.notify()永远不会被调用

Service.getMessages(resultId: self.res![index].id, completionHandler: { (latestMessage, response, error) in

        if let latestMessage = latestMessage {
            self.res![index].messages = [latestMessage]
            print("L2: \(self.res![index].messages)")
            print("C0: \(self.res)")
        }
        group.leave() // continue the loop
    })

ViewController.Swift

var res: [ResultDTO]?

override func viewDidLoad() {
    super.viewDidLoad()

    let group0 = DispatchGroup()

    group0.enter()
    Service.getResults { (results, response, error) in
        guard var results = results else { print("PROBLEM"); return }
        self.res = results
        group0.leave()
    }

    group0.notify(queue: .main) {
        print("in group0")
        let group = DispatchGroup() // initialize
        for index in 0..<self.res!.count {
            group.enter() // wait
            Service.getMessages(resultId: self.res![index].id, completionHandler: { (latestMessage, response, error) in

                if let latestMessage = latestMessage {
                    self.res![index].messages = [latestMessage]
                    print("L2: \(self.res![index].messages)")
                    print("C0: \(self.res)")
                }

            })
            group.leave() // continue the loop
        }

        group.notify(queue: .main) {
            print("In group notify")

            do {
                print("C2: \(self.res)")
                for index in 0..< self.res!.count {
                    print("L4: \(self.res![index].messages)")
                }
                // ... configure my view with the data
            } catch {
                print("Error: \(error)")
            }

        }
    }

}

1 个答案:

答案 0 :(得分:0)

我将通过使用一个调度组来简化您的数据处理(因为您只需要一个调度组):

class YourViewController: UIViewController {

    private var results: [ResultDTO]?

    override func viewDidLoad() {
        super.viewDidLoad()
        getResults()
    }

    private func getResults() {

        Service.getResults { (results, response, error) in

            guard let results = results,
                results.count > 0 else {
                    return
            }

            let dispatchGroup = DispatchGroup()

            for r in results {

                dispatchGroup.enter()
                Service.getMessages(resultId: r.id, completionHandler: { (latestMessage, response, error) in

                    if let latestMessage = latestMessage {
                        r.messages = [latestMessage]
                    }
                    dispatchGroup.leave()

                }

            }

            dispatchGroup.notify(queue: .main, execute: {
                self.results = results
            })

        }

    }

}

仅将此视为起点。如果这是生产代码,我将实现错误处理和后台排队(假设数据在主线程上返回)。