ios – Swift /如何使用dispatch_group与多个称为Web服务的?

我正在使用dispatch_group来调用Firebase请求函数,一旦请求完成,就可以得到通知,以便能够使用结果.在这种情况下,我刚刚放了一个打印声明.
func loadStuff() {
    dispatch_group_enter(group)
        myFirebaseFunction() {
             dispatch_group_leave(group)
        }

    dispatch_group_notify(group,dispatch_get_main_queue()) {
        print("done")
    }
}

func myFirebaseFunction(completionHandler: () -> ()) {

    let usersRef = firebase.child("likes")
    usersRef.observeEventType(.Value,withBlock: { snapshot in

             if snapshot.exists() {
                   let sorted = (snapshot.value!.allValues as NSArray).sortedArrayUsingDescriptors([NSSortDescriptor(key: "date",ascending: false)])

                   for item in sorted {

                       dict.append(item as! NSDictionary)
                   }
            }
            completionHandler()
   })   
}

这段代码正常工作.问题是,在运行时,数据将被添加到Firebase数据库.这就是为什么我必须使用observeEventType而不是observeSingleEventOfType.

这意味着在运行时有一个观察者,如果数据已被添加到数据库中,myFirebaseFunction中的块将再次被调用.

一旦发生这种情况,应用程序崩溃,因为dispatch_group_leave(组)已调用,而没有dispatch_group_enter(组).只要我有这个权利.

dispatch_group_enter(group)
    myFirebaseFunction() {
         dispatch_group_leave(group)      // crash here
    }

如果我将其更改为observeSingleEventOfType,则不会发生崩溃,但是新添加到Firebase的数据将不会被观察到.

在多个运行的Web服务中使用dispatch_group的最佳做法是什么?或者我需要做些什么来解决我的问题?帮助非常感激.

PS目前我正在使用Swift 2.3,但是升级到Swift 3是有计划的,所以收到一个能够兼容的答案将非常棒.

解决方法

问题

正如你所说,调用dispatch_group_enter和dispatch_group_leave必须平衡.在这里,您无法平衡它们,因为执行实时实时获取呼叫的功能离开.

方法1 – 组上的所有呼叫

如果没有问题,myFirebaseFunction总是执行其在该调度组上的工作,那么您可以将它们放在里面,也可以使用beginHandler和completionHandler:

func loadStuff() {
    myFirebaseFunction(beginHandler: {
        dispatch_group_enter(group)
        dispatch_group_notify(group,dispatch_get_main_queue()) {
            print("done")
        }
    },completionHandler: { dispatch_group_leave(group) })

}

func myFirebaseFunction(beginHandler: () -> (),completionHandler: () -> ()) {        

    let usersRef = firebase.child("likes")
    usersRef.observeEventType(.Value,withBlock: { snapshot in

        beginHandler()
        if snapshot.exists() {
            let sorted = (snapshot.value!.allValues as NSArray).sortedArrayUsingDescriptors([NSSortDescriptor(key: "date",ascending: false)])

            for item in sorted {

                dict.append(item as! NSDictionary)
            }
        }
        completionHandler()
   })   
}

在这里,完成处理程序仍然被loadStuff设置为dispatch_group_leave,但是还有一个可以调用dispatch_group_enter和dispatch_group_notify的开始处理程序.通知需要在开始时被调用的原因是,我们需要确保我们在调用notify之前已经进入了组,或者如果组为空,则notify块将立即执行.

传递给dispatch_group_notify的块只会被精确调用一次,即使在通知被调用后在组上执行了块.因此,每次自动调用observeEventType可能会对组发生安全.然后在这些功能之外的任何时候,您需要等待加载完成,您只需调用notify即可.

编辑:因为每次调用beginHandler时都会调用notify,所以这个方法实际上会导致每次调用notify块,所以可能不是一个理想的选择.

方法2 – 只有第一次呼叫组,几种方法

如果您真正需要的仅仅是第一次调用observeEventType来使用该组,那么一个选项是要有两个版本的myFirebaseFunction:一个类似于你已经拥有的版本,一个使用observeSingleEventOfType.然后加载的东西可以调用这两个函数,只有通过dispatch_group_leave作为完成其中之一:

func loadStuff() {
    dispatch_group_enter(group)
        myInitialFirebaseFunction() {
            dispatch_group_leave(group)
        }

    dispatch_group_notify(group,dispatch_get_main_queue()) {
        print("done")
    }

    myFirebaseFunction({})
}

func myInitialFirebaseFunction(completionHandler: () -> ()) {

    let usersRef = firebase.child("likes")
    usersRef.observeSingleEventOfType(.Value,withBlock: { snapshot in
        processSnapshot(snapshot)
        completionHandler()
    })   
}

func myFirebaseFunction(completionHandler: () -> ()) {

    let usersRef = firebase.child("likes")
    usersRef.observeSingleEventOfType(.Value,withBlock: { snapshot in
        processSnapshot(snapshot)
        completionHandler()
    })   
}

func processSnapshot(snapshot: FDataSnapshot) {

    if snapshot.exists() {
        let sorted = (snapshot.value!.allValues as NSArray).sortedArrayUsingDescriptors([NSSortDescriptor(key: "date",ascending: false)])

        for item in sorted {
            dict.append(item as! NSDictionary)
        }
    }
}

方法3 – 只有第一次呼叫组,没有额外的方法

请注意,因为“方法2”中的loadStuff基本上是从Firebase两次加载东西,它可能没有你想要的那么高效.在这种情况下,您可以使用Bool来确定是否应该调用休假:

var shouldLeaveGroupOnProcess = false

func loadStuff() {
    dispatch_group_enter(group)
        shouldLeaveGroupOnProcess = true
        myFirebaseFunction() {
            if shouldLeaveGroupOnProcess {
                shouldLeaveGroupOnProcess = false
                dispatch_group_leave(group)
            }
        }

    dispatch_group_notify(group,withBlock: { snapshot in

        if snapshot.exists() {
            let sorted = (snapshot.value!.allValues as NSArray).sortedArrayUsingDescriptors([NSSortDescriptor(key: "date",ascending: false)])

            for item in sorted {
                dict.append(item as! NSDictionary)
            }
        }
        completionHandler()
    })   
}

这里,即使在初始加载期间进行了对observeEventType的多次调用,也只能保留一次调用,并且不会发生崩溃.

Swift 3

PS Currently I’m using Swift 2.3,but an upgrade to Swift 3 is planned,so it would be very awesome to receive an answer capable for both.

调度已经在Swift 3中进行了彻底的检修(它是面向对象的),所以在两者之间运行良好的代码并不是一件很重要的事情:)

但是上述三种方法的概念是一样的.在Swift 3:

>使用DispatchGroup的其中一个创建您的组> dispatch_group_enter现在是在组上输入的实例方法> dispatch_group_leave现在是组上的实例方法> dispatch_group_notify现在是组上的实例方法通知

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


当我们远离最新的 iOS 16 更新版本时,我们听到了困扰 Apple 最新软件的错误和性能问题。
欧版/美版 特别说一下,美版选错了 可能会永久丧失4G,不过只有5%的概率会遇到选择运营商界面且部分必须连接到iTunes才可以激活
一般在接外包的时候, 通常第三方需要安装你的app进行测试(这时候你的app肯定是还没传到app store之前)。
前言为了让更多的人永远记住12月13日,各大厂都在这一天将应用变灰了。那么接下来我们看一下Flutter是如何实现的。Flutter中实现整个App变为灰色在Flutter中实现整个App变为灰色是非常简单的,只需要在最外层的控件上包裹ColorFiltered,用法如下:ColorFiltered(颜色过滤器)看名字就知道是增加颜色滤镜效果的,ColorFiltered( colorFilter:ColorFilter.mode(Colors.grey, BlendMode.
flutter升级/版本切换
(1)在C++11标准时,open函数的文件路径可以传char指针也可以传string指针,而在C++98标准,open函数的文件路径只能传char指针;(2)open函数的第二个参数是打开文件的模式,从函数定义可以看出,如果调用open函数时省略mode模式参数,则默认按照可读可写(ios_base:in | ios_base::out)的方式打开;(3)打开文件时的mode的模式是从内存的角度来定义的,比如:in表示可读,就是从文件读数据往内存读写;out表示可写,就是把内存数据写到文件中;
文章目录方法一:分别将图片和文字置灰UIImage转成灰度图UIColor转成灰度颜色方法二:给App整体添加灰色滤镜参考App页面置灰,本质是将彩色图像转换为灰度图像,本文提供两种方法实现,一种是App整体置灰,一种是单个页面置灰,可结合具体的业务场景使用。方法一:分别将图片和文字置灰一般情况下,App页面的颜色深度是24bit,也就是RGB各8bit;如果算上Alpha通道的话就是32bit,RGBA(或者ARGB)各8bit。灰度图像的颜色深度是8bit,这8bit表示的颜色不是彩色,而是256
领导让调研下黑(灰)白化实现方案,自己调研了两天,根据网上资料,做下记录只是学习过程中的记录,还是写作者牛逼
让学前端不再害怕英语单词(二),通过本文,可以对css,js和es6的单词进行了在逻辑上和联想上的记忆,让初学者更快的上手前端代码
用Python送你一颗跳动的爱心
在uni-app项目中实现人脸识别,既使用uni-app中的live-pusher开启摄像头,创建直播推流。通过快照截取和压缩图片,以base64格式发往后端。
商户APP调用微信提供的SDK调用微信支付模块,商户APP会跳转到微信中完成支付,支付完后跳回到商户APP内,最后展示支付结果。CSDN前端领域优质创作者,资深前端开发工程师,专注前端开发,在CSDN总结工作中遇到的问题或者问题解决方法以及对新技术的分享,欢迎咨询交流,共同学习。),验证通过打开选择支付方式弹窗页面,选择微信支付或者支付宝支付;4.可取消支付,放弃支付会返回会员页面,页面提示支付取消;2.判断支付方式,如果是1,则是微信支付方式。1.判断是否在微信内支付,需要在微信外支付。
Mac命令行修改ipa并重新签名打包
首先在 iOS 设备中打开开发者模式。位于:设置 - 隐私&安全 - 开发者模式(需重启)
一 现象导入MBProgressHUD显示信息时,出现如下异常现象Undefined symbols for architecture x86_64: "_OBJC_CLASS_$_MBProgressHUD", referenced from: objc-class-ref in ViewController.old: symbol(s) not found for architecture x86_64clang: error: linker command failed wit
Profiles >> 加号添加 >> Distribution >> "App Store" >> 选择 2.1 创建的App ID >> 选择绑定 2.3 的发布证书(.cer)>> 输入描述文件名称 >> Generate 生成描述文件 >> Download。Certificates >> 加号添加 >> "App Store and Ad Hoc" >> “Choose File...” >> 选择上一步生成的证书请求文件 >> Continue >> Download。
今天有需求,要实现的功能大致如下:在安卓和ios端实现分享功能可以分享链接,图片,文字,视频,文件,等欢迎大佬多多来给萌新指正,欢迎大家来共同探讨。如果各位看官觉得文章有点点帮助,跪求各位给点个“一键三连”,谢啦~声明:本博文章若非特殊注明皆为原创原文链接。