- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我循环遍历几个 Url
,将它们转换为 Data
,然后将数据发送到 Firebase Storage
,然后在一切完成后发送收集到 Firebase 数据库
我使用 DispatchGroup() 的 .enter()
开始循环,一旦我将数据发送到 Storage 并获得值 url 字符串 absoluteString
我使用 .leave()
开始下一次迭代。
我意识到在循环过程中,有几个点可能会发生错误:
UrlSession
.putData
函数.downloadURL(completion:...
完成处理程序?.absoluteString
是 nil如果我在这些点中的任何一个点出现错误,我会显示一个警报函数 showAlert(
),它会显示警报并使用 session.invalidateAndCancel()
取消 UrlSession。我取消了所有内容,因为我希望用户重新开始。
由于 DispatchGroup() 在 .enter()
处挂起,我该如何取消 DispatchGroup() 来停止循环?
var urls = [URL]()
var picUUID = UUID().uuidString
var dict = [String:Any]()
let session = URLSession.shared
let myGroup = DispatchGroup()
var count = 0
for url in urls{
myGroup.enter()
session.dataTask(with: url!, completionHandler: {
(data, response, error) in
if error != nil {
self.showAlert() // 1st point of error
return
}
DispatchQueue.main.async{
self.sendDataToStorage("\(self.picUUID)_\(self.count).jpg", picData: data)
self.count += 1
}
}).resume()
myGroup.notify(queue: .global(qos: .background) {
self.sendDataFromDictToFirebaseDatabase()
self.count = 0
self.session.invalidateAndCancel()
}
}
func sendDataToStorage(_ picId: String, picData: Data?){
dict.updateValue(picId, forKey:"picId_\(count)")
let picRef = storageRoot.child("pics")
picRef.putData(picData!, metadata: nil, completion: { (metadata, error) in
if error != nil{
self.showAlert() // 2nd point of error
return
}
picRef?.downloadURL(completion: { (url, error) in
if error != nil{
self.showAlert() // 3rd point of error
return
}
if let picUrl = url?.absoluteString{
self.dict.updateValue(picUrl, forKey:"picUrl_\(count)")
self.myGroup.leave() //only leave the group if a Url string was obtained
}else{
self.showAlert() // 4th point of error
}
})
})
}
func showAlert(){
// the DispatchGroup() should get cancelled here
session.invalidateAndCancel()
count = 0
UIAlertController...
}
func sendDataFromDictToFirebaseDatabase(){
}
最佳答案
在问题下方的评论中,@rmaddy 说“无论成功与否,您都需要调用 leave
”。我这样做了,但循环仍在运行并且 sendDataFromDictToFirebaseDatabase()
仍然触发事件,尽管出现错误。
我能找到的唯一解决方法是将循环放在一个带有完成处理程序的函数中,并使用 bool
来决定 sendDataFromDictToFirebaseDatabase()
是否应该火:
var urls = [URL]()
var picUUID = UUID().uuidString
var dict = [String:Any]()
let session = URLSession.shared
let myGroup = DispatchGroup()
var count = 0
var wasThereAnError = false // use this bool to find out if there was an error at any of the error points
func loopUrls(_ urls: [URL?], completion: @escaping ()->()){
for url in urls{
myGroup.enter()
session.dataTask(with: url!, completionHandler: {
(data, response, error) in
if error != nil {
self.showAlert() // 1st point of error. If there is an error set wasThereAnError = true
return
}
DispatchQueue.main.async{
self.sendDataToStorage("\(self.picUUID)_\(self.count).jpg", picData: data)
self.count += 1
}
}).resume()
myGroup.notify(queue: .global(qos: .background) {
completion()
}
}
}
// will run in completion handler
func loopWasSuccessful(){
// after the loop finished this only runs if there wasn't an error
if wasThereAnError == false {
sendDataFromDictToFirebaseDatabase()
count = 0
session.invalidateAndCancel()
}
}
func sendDataToStorage(_ picId: String, picData: Data?){
dict.updateValue(picId, forKey:"picId_\(count)")
let picRef = storageRoot.child("pics")
picRef.putData(picData!, metadata: nil, completion: { (metadata, error) in
if error != nil{
self.showAlert() // 2nd point of error. If there is an error set wasThereAnError = true
return
}
picRef?.downloadURL(completion: { (url, error) in
if error != nil{
self.showAlert() // 3rd point of error. If there is an error set wasThereAnError = true
return
}
if let picUrl = url?.absoluteString{
self.dict.updateValue(picUrl, forKey:"picUrl_\(count)")
self.myGroup.leave() // leave group here if all good on this iteration
}else{
self.showAlert() // 4th point of error. If there is an error set wasThereAnError = true
}
})
})
}
func showAlert(){
wasThereAnError = true // since there was an error set this to true
myGroup.leave() // even though there is an error still leave the group
session.invalidateAndCancel()
count = 0
UIAlertController...
}
func sendDataFromDictToFirebaseDatabase(){
}
并使用它:
@IBAction fileprivate func postButtonPressed(_ sender: UIButton) {
wasThereAnError = false // set this back to false because if there was an error it was never reset
loopUrls(urls, completion: loopWasSuccessful)
}
关于ios - Swift iOS - 如何取消 DispatchGroup() 管理循环,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/50478833/
我想知道以下情况。 我在一个函数中有一个 DispatchGroup。现在我在后台线程上进入组并调用 wait()。 func test() { let group = DispatchGrou
下面的代码我用来进行并发 API调用。不知怎的,这个方法返回多次。我已经在没有 DispatchGroup 的情况下进行了测试,它按预期工作。帮我找出为什么它多次调用。 我的代码片段: fu
是否可以从另一个类访问 DispatchGroup?假设我在类 Loginfunctions.swift 中有以下函数(在 func 中定义了 var loginQueue = DispatchGro
我正在尝试使用 Firebase 身份验证来注册用户。当用户注册时,我希望将他们添加到我的 Users 中Firestore 中的集合以及 Users授权部分。 createUser(withEmai
我正在尝试链接两个调度组,但我的代码总是在随机位置崩溃。 我的功能是这样的: func getAllActivities(userUID: String, _ completionHandler: @
我想使用 DisaptchGroup 在遍历并将元素追加到数组后调用 func。 这是 firebase 节点的样子 { "tags": { "NewYork":
enter code here我用调度组尝试了很多东西,但我无法获得稳定的结果。自从我的服务器以来,我使用 Alamofire 获取数据。我在 Helper 类中编写了一个函数,并在 AppDeleg
我目前有一个数组,它遍历图像路径的值,并希望在第一个 for 循环中异步检索这些图像。我试过使用两个调度组,但总是在检索图像之前返回完成处理程序。 static func getAllEntriesW
我创建了一个 DispatchGroup 并运行了 2 个异步任务。一个在 main 上,另一个在 global() 上。 据我所知,DispatchGroup.notify 的 block 应该在所
因此,我花了一些时间尝试让 DispatchGroup 在长时间的异步操作完成之前阻止 for 循环迭代。我发现的大多数示例都相当简单明了,但我似乎无法让我的简单测试用例按预期工作。 let grou
我有两个函数(或任务),我想一个接一个地运行,我正在使用 DispatchGroup 来跟踪它们并在它们完成时通知我。现在它们正在主线程中完成,但我想在后台线程中运行这些任务。我将如何去做呢?我尝试了
以下是否会产生可能的竞争条件? let syncGroup = DispatchGroup() var tempData: [Int] for index in 1...10 { sync
我必须停止我的代码一秒钟,以便在继续之前同步服务器数据库。 下面的所有代码片段都是从主线程运行的。 我首先使用了这个: // code 1 DispatchQueue.main.asyncAfter(
我正在进行两个异步网络调用,并且想使用一个 Dispatch Group 来等待调用完成然后恢复。我的程序卡住了。 class CommentRatingViewController: UIViewC
我想让对服务的调用同步,因为我希望我的调用返回已使用泛型映射的对象。 这是代码: func execute(request: HttpRequest, responseType: T.Type) th
我遇到一种情况,我需要等待一组任务完成才能执行某些 UI 渲染代码。 其中一些任务是网络请求,它们始终会完成并离开调度组。然而,有些不是网络请求,可能会也可能不会完成。如果一项任务未完成,并且离开的调
在使用异步递归调用 API 时,我无法理解如何使用 GCD。 下面是三个类似方法之一,包含相同的逻辑,只是针对不同的数据和 API 端点。如果没有下一页请求,该方法应该完成并且下一个方法应该开始。 我
我在理解或使用 Dispatchgroup 时遇到问题。我已经阅读了很多关于它们的内容,但是大多数示例/文档都非常模糊或者与我想做的不一样,但是每次我提到我的问题时,每个人都说“使用调度组!”。 这是
问题描述: 我想通过“DispatchGroup”执行一系列异步任务,当所有这些任务完成时它返回结果。另外,我想设置限制进程的超时时间,到时候把成功的结果发给我。我使用了以下结构: 代码块 let m
在下面的代码中,附加到数组是否安全?是否保证维持秩序? let processedData: [SomeType] = [] let dispatchGroup = DispatchGroup() f
我是一名优秀的程序员,十分优秀!