我是斯威夫特的新手,正在练习社交。
我有一个孩子的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。一般来说,你不应该这样做。您应该使用查询。但这就是如何在需要时将异步函数转换为同步函数的方法。