如何在异步调用完成后运行代码,代码在swift3

时间:2017-04-11 00:33:11

标签: ios swift completionhandler

我有一个包含帖子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)
}

1 个答案:

答案 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)
    }
}