关于RxSwift通知中心NotificationCenter的一个bug

问题描述

  1. 列表中的任务,是要用户分享到微信群或者朋友圈,当分享成功收到回调之后,把改任务设置成"已完成"
  2. 连续若干次分享,前面n次都没有分享成功,第n+1次分享成功后,前面的所有任务都变成了"已完成"

问题排查

在相关页面中,有个shareToWechat的方法

当用户点击了"分享"按钮,就调用这个方法,方法中添加了一个微信分享回调通知监听

监听回调方法中,把该任务id加入到已完成列表,代码如下:

// 分享到微信
    private func shareToWechat(model: GWXShareModel, task: GOwnMissionMetaTask){
        // 调用分享
        if wxShareSendMultiMessage(model: model) == false {
            SVProgressHUD.showError(withStatus: "分享失败")
            return
        }
        SVProgressHUD.dismiss()
        NotificationCenter.default.rx
            .notification(Notification.Name.init(R.string.global.kNoti_Share_Wechat_Success()))
            .subscribe({ (noti) in
            print("~~~~分享成功!!")
            SVProgressHUD.dismiss()
            // 把当前任务加到完成列表里,用户可以领取奖励
            self.viewModel.addTaskToDoneList(id: task.id)
            // 刷新列表
            self.viewModel.missionList.value = self.viewModel.missionList.value
            
        }).disposed(by: self.disposeBag)
    }

问题分析

  • 通过断点调试,发现当第n+1次分享成功后,发射了一次成功回调的广播

  • 成功的回调被注册了n+1次,于是回调方法调用了n+1次,问题就出在了这里.

  • 所以 问题的关键是: 如何保证没有分享成功的任务,不执行成功的回调方法

问题解决

为了达到目的,我目前有两个想法:

  1. 在分享成功的广播发送后,分别进入到n+1个回调操作,然后通过某种条件判断当前的任务是否成功
  2. 在分享失败的广播发送后,把对任务成功的订阅事件销毁掉

经过一番思索,发现想法1不可行,想法2很容易做到,最终的代码如下:

    // 分享到微信
    private func shareWX(model: GWXShareModel, task: GOwnMissionMetaTask){
        // 调用分享
        if wxShareSendMultiMessage(model: model) == false {
            SVProgressHUD.showError(withStatus: "分享失败")
            return
        }
        // 分享成功的监听
        let notificationSubscription = NotificationCenter.default.rx
            .notification(Notification.Name.init(R.string.global.kNoti_Share_Wechat_Success()))
            .subscribe({ (noti) in
            print("~~~~分享成功!!")
            SVProgressHUD.dismiss()
            // 把当前任务加到完成列表里,用户可以领取奖励
            self.viewModel.addTaskToDoneList(id: task.id)
            // 刷新列表
            self.viewModel.missionList.value = self.viewModel.missionList.value
        })
        // 分享失败的监听
        NotificationCenter.default.rx
        .notification(Notification.Name.init(R.string.global.kNoti_Share_Wechat_Fail()))
            .subscribe { (noti) in
                // 微信分享失败后,把对分享成功的rx监听对象销毁
                notificationSubscription.dispose()
        }.disposed(by: disposeBag)
    }

总结

对于监听通知,要留意是否注册了多余的订阅者,如果不及时清除很可能会有bug

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 1.ios高性能编程 (1).内层 最小的内层平均值和峰值(2).耗电量 高效的算法和数据结构(3).初始化时...
    欧辰_OSR阅读 29,645评论 8 265
  • 值此平安夜、圣诞、元旦来临之际,现邀请各位亲朋好友前来浦东相聚,借此以分享我再次喜得千金之快乐! 另附爱女照...
    萌豆宝阅读 338评论 0 0
  • 子曰:父在观其志,父没观其行。三年无改于父之道,可谓孝矣。 感悟:钱穆老先生说,论语文辞简约,异解逐滋。的确如此!...
    名言名语March阅读 127评论 0 0
  • 稻盛和夫指出,作为领导者,应该保持积极的心态,其中包括付出超过常人的努力,取得下属的认同感等。 “领导者是一个部门...
    金尚沄阅读 11,871评论 0 0
  • 轻风细雨乍晓寒。草枯叶落,已觉秋心难。问声雁儿为何单?半杯残酒痴相缠。 轻解罗衫挥袖舞。斜靠柳肩,枕风...
    爱那么多阅读 338评论 0 3