从异步调用修改数组不修改数组 swift
Modifying array from asynchronous call not modifying the array swift
我正在尝试使用另一个异步调用的结果来修改从一个异步调用中检索到的对象数组。基本上我检索了一个结果数组,并且有一个字段 messages
从我的服务器调用返回 nil 。然后,我需要在 for 循环中使用每个 result
的 id 进行另一个服务器调用,以获取 messages
数组。然后我将 messages
设置为返回值。
大约 20 分钟前,我刚刚从 那里了解到如何使用 DispatchGroup(),所以请耐心等待。
我遇到的问题是,当我执行 print("C0: (self.res)") 时,它会打印出所有项目,包括我刚刚添加的 messages
。当 print("C2: (self.res)") 在 group.notify(...)
中打印时,所有 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)")
}
}
}
}
我会使用一个调度组来简化您的数据处理(因为您只需要一个):
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
})
}
}
}
这只是一个起点。如果这是生产代码,我会实施错误处理和后台排队(假设数据在主线程上返回)。
我正在尝试使用另一个异步调用的结果来修改从一个异步调用中检索到的对象数组。基本上我检索了一个结果数组,并且有一个字段 messages
从我的服务器调用返回 nil 。然后,我需要在 for 循环中使用每个 result
的 id 进行另一个服务器调用,以获取 messages
数组。然后我将 messages
设置为返回值。
大约 20 分钟前,我刚刚从
我遇到的问题是,当我执行 print("C0: (self.res)") 时,它会打印出所有项目,包括我刚刚添加的 messages
。当 print("C2: (self.res)") 在 group.notify(...)
中打印时,所有 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)")
}
}
}
}
我会使用一个调度组来简化您的数据处理(因为您只需要一个):
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
})
}
}
}
这只是一个起点。如果这是生产代码,我会实施错误处理和后台排队(假设数据在主线程上返回)。