Swift – 使用两个不同的OperationQueue和KVO时应用程序崩溃

Swift – 使用两个不同的OperationQueue和KVO时应用程序崩溃,第1张

概述我正在使用 JSON获取两种类型的信息,并且我使用addObserver(forKeyPath:“operations”…)将“ *** 作”添加到2个不同的 *** 作队列类中. 在函数observeValue中,我正在检查operationQueue1.operations.isEmpty,然后我在UI中刷新我的信息.我正在使用operationQueue2执行相同的 *** 作,但是当有时启动2个 *** 作时,应用程序崩 我正在使用 JSON获取两种类型的信息,并且我使用addobserver(forKeyPath:“operations”…)将“ *** 作”添加到2个不同的 *** 作队列类中.
在函数observeValue中,我正在检查operationQueue1.operations.isEmpty,然后我在UI中刷新我的信息.我正在使用operationQueue2执行相同的 *** 作,但是当有时启动2个 *** 作时,应用程序崩溃并显示错误消息:***由于未捕获的异常’NSRangeException’而终止应用程序,原因:’无法删除观察者&lt ; Appname.VIEwController 0x102977800>对于来自< Appname.OperationQueue1 0x1c4a233c0>的关键路径“ *** 作”.因为它没有注册为观察员.
只有1个 *** 作开始时我没有问题.有什么建议?
func getInfo1(){//runned in vIEwDIDLoad  operationQueue1.addobserver(forKeyPath:"operations"...)  operationQueue1.dataTask(URL:"..."....){      dispatchQueue.main.async{  NotificationCenter.default.postNotification(NSNotification.name(rawValue: "NewDataReceived1",userInfo:infoFromTheWebsite)      }  }}func NewDataReceived1(){  here I add the information to arrays to be loaded in tableVIEw1}HERE IS THE CODE FOR 2ND INFO WHICH IS THE SAMEoverrIDe func observeValue(forKeyPath keyPath: String?,....){        if(object as? operationQueue1 == operationQueue1Class && keyPath == "operations" && context == context1){             if(operationQueue1.operations.isEmpty){                  dispatchQueue.main.async{                        operationQueue1..removeObserver(self,forKeyPath:"operations")                        Timer.scheduled("refreshingtableinformation1")                    }             }        }else if(operationQueue2....){             SAME AS OPERATION 1,BUT USING DIFFERENT FUNC TO REFRESH table informatION AND THE tableS ARE DIFFERENT        }else{            super.observeValue(forKeyPath: keyPath,of: object,change: change,context: context)        }}func refreshingtableinformation1(){     tableVIEw1.reloadData()     Timer.scheduled("getInfo1",repeat:false)}func refreshingtableinformation2(){     tableVIEw2.reloadData()     Timer.scheduled("getInfo2",repeat:false)}

有时候它可以工作10秒钟并且崩溃,有时可以工作超过60秒然后崩溃……

您的代码易受竞争条件的影响.请考虑以下情形:

>调用getInfo1(),它向operationQueue1添加一个 *** 作.
> *** 作完成,这意味着您的KVO观察被调用.队列现在为空,因此您的观察计划会在主调度队列中删除您的观察者.
>现在,在您提交到主队列的 *** 作之前能够运行,其他东西调用getInfo1(),这会向operationQueue1添加一个新 *** 作,该 *** 作在您在步骤2中排队的 *** 作有机会运行之前完成(嘿,也许主要队列忙于某事;这很容易发生,因为它是一个串行队列).
>当队列为空时,再次调用对第一次调用getInfo1()的观察,导致另一个注销块被提交到主队列.
>最后两个deregister块在主队列上执行.第二个崩溃程序,因为你已经注销了你的观察者.

您可以通过使用Swift 4的基于块的观察者来修复此问题(假设代码没有此类问题的更多问题),并将观察者设置为nil而不是显式取消注册.但是,我建议KVO是你想要做的wrong tool.正如旧的“水晶探索”游戏的说明所说,这有点像使用高射炮杀死蚊子.

从我上面的代码中可以看出,看起来你正在使用KVO来安排一个通知,告知你提交给队列的 *** 作或一组 *** 作何时完成.根据您的dataTask方法实际执行的 *** 作,以下是我要做的事情:

>如果只提交一个 *** 作:将 *** 作的completionBlock属性设置为一个刷新表信息的闭包.
>如果您提交了多个 *** 作:创建一个新的BlockOperation,刷新您的表信息,并使用您提交给队列的每个其他 *** 作调用该 *** 作的addDependency.然后,提交该 *** 作.

这将为您提供更清洁,更无故障的方法来监控您的任务完成情况.而且由于您不再需要队列完全清空,您甚至可能不再需要使用两个单独的队列,具体取决于您使用它们做了什么.

总结

以上是内存溢出为你收集整理的Swift – 使用两个不同的OperationQueue和KVO时应用程序崩溃全部内容,希望文章能够帮你解决Swift – 使用两个不同的OperationQueue和KVO时应用程序崩溃所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/web/1029894.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-05-23
下一篇 2022-05-23

发表评论

登录后才能评论

评论列表(0条)

保存