我是斯威夫特的新手,正在练习社交。
我有一个孩子的ID列表。完整的孩子信息存储在Firebase数据库中。数据库中的每个孩子都有一个ID和一个属性isGamer: Bool
我想创建一个函数,它将返回第一个孩子isGamer==true.
我想的逻辑是:
for循环,它将在列表中运行,如果isGamer == true则检查第一个孩子,如果是,则返回孩子的id,否则继续搜索。
我想这是可行的,尽管我不知道如何处理所有这些异步的东西,如何使等待响应循环等等。
主要功能将管理所有内容:

// Mother Function
func printFirstGamer(kidsIds: [String]) {
     print(returnFirstKidIsGamer(kidsIds: kidsIds ))
}

另一个函数将查找isgamer==true的第一个孩子:
// This is where I'm stuck basically

func returnFirstKidIsGamer(kidsIds: [String]) -> String {
    for kidID in kids {
        // Wait for previous request to finish before trying again.
        DataService.instance.isKidAGamer(kidID: kidID) { (isGamer) in

        }
        // The for loop will wait for response and then if isGamer==false, will continue, if isGamer==true will return
    }
    // return first kid where isGamer == true

}

我将使用此功能检查每个孩子的isgamer:
func isKidAGamer(kidID: String, completed: @escaping isKidAGamerCompleted) {
        mainRef.child("Kids").child(kidID).child("isGamer").observeSingleEvent(of: .value, with: { (result) in
            if result.exists() {
                let isGamer = result.value as! Bool
                completed(isGamer)
            } else {
                completed(false)
            }
        })
    }

注意:我不想使用查询,我正在练习(:
谢谢(:

最佳答案

使用dispatchgroup可以使此同步,但决不能在主队列上调用它。

// Blocking function. Must not be called on main queue!
func returnFirstKidIsGamer(kidsIds: [String]) -> String? {
    let group = DispatchGroup()
    var result: String? = nil

    for kidID in kidsIds {
        // Wait for previous request to finish before trying again.
        group.enter()
        DataService.instance.isKidAGamer(kidID: kidID) { (isGamer) in
            if isGamer {
                result = kidId
            }
            group.leave()
        }
        group.wait()
        guard result == nil else { break }
    }
    return result
}

在进入每个循环之前调用group.enter(),在每个步骤完成时调用group.leave()。然后等待步骤完成后再继续。
这个函数是同步的。它会阻塞队列,因此决不能在主队列上调用它。你必须把它移到背景上,像这样:
DispatchQueue.global(qos: .userInitiated).async {
    let kidId = returnFirstKidIsGamer(kidsIds: [kids])
    DispatchQueue.main.async {
        doSomethingInTheUIWithValue(kidId)
    }
}

注意,这将返回String?,而不是String,因为找不到id。
一般来说,你不应该这样做。您应该使用查询。但这就是如何在需要时将异步函数转换为同步函数的方法。

09-26 01:00