我有一个包含帖子ID的数组,然后我做一个for
循环并检索每个帖子内容并将帖子附加到帖子数组。循环结束后,我需要在完成处理程序中返回post数组。
但是,.observeSingleEvent是一个异步方法,因此从Firebase检索帖子内容,已经执行了完成处理程序(posts),返回一个空数组。我想在每个帖子附加到数组后返回数组,类似于使这些方法同步。
static func getPost(postKeys:[String], completionHandler: @escaping ([Post])->()){
var posts = [Post]()
for postKey in postKeys{
let postRef = DataService.ds.REF_POSTS.child(postKey)
postRef.observeSingleEvent(of: .value, with: { (snapshot) in
if let postDict = snapshot.value as? Dictionary<String, Any>{
//create a post and append it to the post array
let key = snapshot.key
let post = Post(postKey: key, postData: postDict)
posts.append(post)
}
})
}
completionHandler(posts)
}
答案 0 :(得分:3)
一系列任务完成后,您可以使用DispatchGroup
执行某些代码。
在开始任务之前调用enter
函数,在任务完成后调用leave
函数。您可以使用notify
提供一个关闭,该关闭将在所有任务“离开”DispatchGroup
后执行。
您还需要小心避免更新阵列时出现并发问题; Swift数组不是线程安全的。在串行调度队列上执行阵列更新可以解决此问题。
static func getPost(postKeys:[String], completionHandler: @escaping ([Post])->()){
var posts = [Post]()
let dispatchGroup = DispatchGroup()
let arrayQueue = DispatchQueue(label: "arrayQueue")
for postKey in postKeys{
dispatchGroup.enter()
let postRef = DataService.ds.REF_POSTS.child(postKey)
postRef.observeSingleEvent(of: .value, with: { (snapshot) in
if let postDict = snapshot.value as? Dictionary<String, Any>{
//create a post and append it to the post array
let key = snapshot.key
let post = Post(postKey: key, postData: postDict)
arrayQueue.sync {
posts.append(post)
}
}
dispatchGroup.leave()
})
}
dispatchGroup.notify(queue: DispatchQueue.main) {
completionHandler(posts)
}
}