gpt4 book ai didi

ios - 在 Swift 中更新变量之前数组正在更新

转载 作者:行者123 更新时间:2023-11-28 07:19:08 25 4
gpt4 key购买 nike

我正在尝试从 Firestore 获取玩具列表并将其放入数组中但是当我调用函数时,它返回空数组,并且在返回后就打印 Toy 对象,因此顺序被破坏了。

我认为闭包对我有帮助,但我想我不知道如何使用它们,而且来自 Google 的示例对我没有帮助

这是我的代码(我使用 SwiftUI,所以我创建了带变量的 swift 文件)

let db = Firestore.firestore()
class DataLoade {
func loadFirebase(completionHandler: @escaping (_ toys: [Toy]) -> ()){
var toysar: [Toy] = []
let toysRef = db.collection("Toys")
toysRef.getDocuments() { (querySnapshot, err) in
if let err = err {
print("Error getting documents: \(err)")
} else {
for document in querySnapshot!.documents {
var name: String = document.get("name") as! String
var id: Int = document.get("id") as! Int
var description: String = document.get("description") as! String
var imageName: String = document.get("imageName") as! String
var price: String = document.get("price") as! String
var category: String = document.get("category") as! String
var timeToy = Toy(id: id, name: name, imageName: imageName, category: category, description: description, price: price)
toysar.append(timeToy)


}




}
}

completionHandler(toysar)
// print(toysar)

}




}


这就是它打印出来的内容:

[] // it prints empty array, but it  is in the end of the code
Toy(id: 1001, name: "Pikachu", imageName: "pikachu-plush", category: "lol", description: "kek", price: "350₽") // and now it prints Toy object, however it is in the start of the code

好的,所以我尝试为我的函数创建完成处理程序,就像在“重复”答案中一样,但这不起作用:数组在完成处理程序工作之前返回

ContentView.swift  
func updateArray() -> [Toy]{
dl.loadFirebase() { toys in
ll = toys

}
print("lol \(datas)") // prints «lol []»
return ll
}

最佳答案

您可以使用 DispatchGroup 等待异步任务。但诀窍是不要将异步任务与 return 语句相关联。相反,使用 closures 在任务完成后执行操作。免责声明:我在 SO 上写了这篇文章,我提前为语法问题道歉。

let toyData = loadFirebase( { (toys) in
print(toys)
//Do something with toys when done
//You could add another completionHandler incase it fails.
//So 1 for pass and 1 for fail and maybe another for cancel. W/e u want
} )
let db = Firestore.firestore()

func loadFirebase(completionHandler:@escaping ((toys: [Toy]?) -> Void)) {
//Create Group
let downloadGroup = DispatchGroup()
var toysar: [Toy] = []
let toysRef = db.collection("Toys")
//If you had multiple items and wanted to wait for each, just do an enter on each.
downloadGroup.enter()
toysRef.getDocuments() { (querySnapshot, err) in
if let err = err {
print("Error getting documents: \(err)")
} else {
for document in querySnapshot!.documents {
var name: String = document.get("name") as! String
var id: Int = document.get("id") as! Int
var description: String = document.get("description") as! String
var imageName: String = document.get("imageName") as! String
var price: String = document.get("price") as! String
var category: String = document.get("category") as! String
var timeToy = Toy(id: id, name: name, imageName: imageName, category: category, description: description, price: price)
toysar.append(timeToy)
print(timeToy)
}
//We aren't done until AFTER the for loop, i.e., each item is grabbed.
downloadGroup.leave()
}
}
//Once the queue is empty, we notify the queue we are done
downloadGroup.notify(queue: DispatchQueue.main) {
completionHandler(toys)
}
}
import SwiftUI
var dl = DataLoade()
var ll: [Toy] = []

let semaphore = DispatchSemaphore(value: 1)
struct ContentView: View {
var items: [Toy]
var body: some View {

NavigationView{
ScrollView(){
VStack(alignment: .leading){
ToyRow(category: "Наш выбор", toys: items)

Spacer()
ToyRow(category: "Акции", toys: items)


}
}.navigationBarTitle(Text("Игрушки г.Остров"))}


}
}
func upe(completionHandler:@escaping ((toys: [Toy]?){
dl.loadFirebase(completionHandler: { toy in
ll.append(contentsOf: toy!)
completionHandler(ll)
} )
}
struct ContentView_Previews: PreviewProvider {
static var previews: some View {
upe(completionHandler: { (toys) in
DispatchQueue.main.async {
ContentView(items: toys)
}
})
}
}

关于ios - 在 Swift 中更新变量之前数组正在更新,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58650808/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com