RxSwift-内存管理

一、循环引用

  • weak:弱引用,[weak self],需要解包操作,延迟调用为nil不崩溃
  • unowned:无主引用,[unowned self],不需要解包操作,延迟调用崩溃
    func retainCycleDemo() {
//        myClosure = {
//            print(self)
//        }
        
        //[weak self]
        //需要解包操作
//        myClosure = { [weak self] in
//            print(self!)
//        }
        
        //[unowned self]
        //不需要解包操作
//        myClosure = { [unowned self] in
//            print(self)
//        }

        //延迟调用
        // nil,不崩溃
//        myClosure = { [weak self] in
//            DispatchQueue.global().asyncAfter(deadline: .now()+2, execute: {
//                print(self!)
//            })
//        }

        //延迟调用
        //崩溃
        myClosure = { [unowned self] in
            DispatchQueue.global().asyncAfter(deadline: .now()+2, execute: {
                print(self)
            })
        }
        
        self.myClosure!()
    }
    
    deinit {
        print(self,"走了")
    }
  • 延迟调用解决,只能用weak,不能用unowned。
  • strongSelf可以换成Self,Self在10.2的时候移除了保留关键字,你可以使用
  • strongSelf 可选绑定之后,不需要再解包
        myClosure = { [weak self] in
            guard let strongSelf = self else { return }
            DispatchQueue.global().asyncAfter(deadline: .now()+2, execute: {
                print(strongSelf)
            })
            
//            guard let self = self else { return }
//            DispatchQueue.global().asyncAfter(deadline: .now()+2, execute: {
//                print(self)
//            })
        }
  • self 只是一个变量
        //只有可选类型,才可以使用可选绑定
        let number: Int? = 12
        if let num = number {
            print(num)
        }
        if let self = number {
            print(self)
        }

12
12

二、RxSwift.Resources.total

  • Podfile 中添加下列代码
  • pod update
  • 如果还是不行,退出Xcode,重新build,重新run
post_install do |installer|
  installer.pods_project.targets.each do |target|
    if target.name == 'RxSwift'
      target.build_configurations.each do |config|
        if config.name == 'Debug'
          config.build_settings['OTHER_SWIFT_FLAGS'] ||= ['-D', 'TRACE_RESOURCES']
        end
      end
    end
  end
end

三、

  • 产生循环引用,解决 [weak self]
        self.accountTF.rx.text.orEmpty
            .subscribe(onNext: { [weak self](text) in
                self?.title = text
            })
            .disposed(by: disposeBag)

  • .debug()
        self.accountTF.rx.text.orEmpty
            .debug()
            .subscribe(onNext: { [weak self](text) in
                self?.title = text
            })
            .disposed(by: disposeBag)
  • 作为参数传递进去,不会产生循环引用
        self.accountTF.rx.text.orEmpty
            .bind(to: self.rx.title)
            .disposed(by: disposeBag)

四、self.observable持有序列

  • 持有序列 - create,产生循环引用,解决[weak self]
  • 持有序列 - 订阅,产生循环引用,解决[weak self]
        // 持有序列 - create
        // self -> observable -> create{} -> self
        self.observable = Observable<Any>.create { [weak self] (anyObserver) -> Disposable in
            anyObserver.onNext("Hello word")
            print(self!)
            return Disposables.create()
        }

        // 持有序列 - 订阅
        // self -> observable -> subscribe onNext -> self
        self.observable?.subscribe( onNext: { [weak self] in
            print(self!)
            print("订阅到1:\($0) --")
        })
            .disposed(by: self.disposeBag)

五、self.observer = anyObserver

  • create中不会产生循环引用。create -> self -> AnyObserver
  • onNext中会产生循环引用,解决[weak self]。create -> self -> AnyObserver -> observer(AnonymousObserver) -> AnonymousObservableSink.on -> AnonymousObserver.on -> onNext?(value) -> subscribe{} -> self
        Observable<Any>.create { (anyObserver) -> Disposable in
            self.observer = anyObserver
            anyObserver.onNext("Hello word")
            print(self)
            return Disposables.create()
            }
            .subscribe(onNext: { [weak self](item) in
                print(self)
                print("订阅到:\(item)")
            })
            .disposed(by: self.disposeBag)

六、订阅中只要使用了self,就一定会产生循环引用

1、sink -> 观察者 -> 订阅的闭包 -> self - 间接持有
2、sink -> disposer -> sink - 响应式关系无法释放 -> 导致 循环引用 -> 计数有问题
3、响应式关系的销毁是在deinit中
4、self 想要释放是要没有东西对它持有,但是1对它间接持有了,sink释放不掉,self又怎么能释放

        _ = Observable<Any>.create { (anyObserver) -> Disposable in
            anyObserver.onNext("Hello word")
            return Disposables.create()
            }
            .subscribe(onNext: { [weak self](item) in
                print(self)
                print("订阅到:\(item)")
            })
            .disposed(by: disposeBag)

七、传值计数问题

问题一、disposeBag的销毁是要等到self销毁的时候销毁
        let vc = LGDetialViewController()
        vc.publicOB.subscribe(onNext: { (item) in
            print("订阅到 \(item)")
        })
        .disposed(by: disposeBag)//disposeBag的销毁是要等到self销毁的时候销毁
class LGDetialViewController: UIViewController {

    let disposeBag = DisposeBag()

    fileprivate var mySubject = PublishSubject<Any>()
    var publicOB : Observable<Any>{
        return mySubject.asObservable()
    }
    
    override func viewWillAppear(_ animated: Bool) {
        super.viewWillAppear(animated)
        print("*****LGDetialViewController出现了:RxSwift的引用计数: \(RxSwift.Resources.total)")
        print("****************************************")
    }
}
  • 方法一:vc.disposeBag
        let vc = LGDetialViewController()
        vc.publicOB.subscribe(onNext: { (item) in
            print("订阅到 \(item)")
        })
        .disposed(by: vc.disposeBag)//vc.disposeBag的销毁是要等到vc销毁的时候销毁
  • 方法二:.takeUntil(vc.rx.deallocated)
        let vc = LGDetialViewController()
        _ = vc.publicOB.takeUntil(vc.rx.deallocated)
            .subscribe(onNext: { (item) in
                print("订阅到 \(item)")
            })
  • 方法三:发送mySubject.onCompleted(),在LGDetialViewController的viewWillDisappear中
        //方法三:发送mySubject.onCompleted(),在LGDetialViewController的viewWillDisappear中
        let vc = LGDetialViewController()
        _ = vc.publicOB
            .subscribe(onNext: { (item) in
                print("订阅到 \(item)")
            })

class LGDetialViewController: UIViewController {
    override func viewWillDisappear(_ animated: Bool) {
        super.viewWillDisappear(animated)
        mySubject.onCompleted()
    }
}
问题二:属性持有 let vc = LGDetialViewController(),导致进去发送onNext("dyz"),再回来(只回到持有LGDetialViewController的VC,再往后回的话就会导致持有LGDetialViewController的VC释放,最后就都释放了,所以也就不会有后面的问题了),再进去没法发送onNext("dyz")消息了。
        _ = vc.publicOB
            .subscribe(onNext: { (item) in
                print("订阅到 \(item)")
            })
class ViewController: UIViewController {

    let disposeBag = DisposeBag()
    let vc = LGDetialViewController()
}
  • 解决方法
  • 重新初始化 mySubject = PublishSubject<Any>()
  • 发送mySubject.onCompleted(),在LGDetialViewController的viewWillDisappear中
class LGDetialViewController: UIViewController {
    fileprivate var mySubject = PublishSubject<Any>()
    var publicOB : Observable<Any>{
        mySubject = PublishSubject<Any>()
        return mySubject.asObservable()
    }

    override func viewWillDisappear(_ animated: Bool) {
        super.viewWillDisappear(animated)
        mySubject.onCompleted()
    }

    override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {
        mySubject.onNext("dyz")
    }
}

八、KVO

不管是KVO还是其他的一定都要加入到disposeBag中,否则就会导致释放不干净,可用RxSwift.Resources.total检测

九、Cell.button响应出现混乱

方法一、没有复用,不好
    override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {

        let cell = tableView.dequeueReusableCell(withIdentifier: "reuseIdentifier", for: indexPath) as! LGTableViewCell        
        cell.disposeBag = DisposeBag()
        cell.button.rx.tap
            .subscribe(onNext: { () in
                print("点击了 \(indexPath)")
            }).disposed(by: cell.disposeBag)
        
        return cell
    }
方法二、有复用,推荐使用

1、.disposed(by: cell.disposeBag),添加cell的disposeBag
2、在Cell的prepareForReuse中销毁cell的disposeBag

    override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let cell = tableView.dequeueReusableCell(withIdentifier: "reuseIdentifier", for: indexPath) as! LGTableViewCell
        cell.button.rx.tap
            .subscribe(onNext: { () in
                print("点击了 \(indexPath)")
            }).disposed(by: cell.disposeBag)
        return cell
    }
class LGTableViewCell: UITableViewCell {

    var button:UIButton!
    var disposeBag = DisposeBag()
    
    override func prepareForReuse() {
        super.prepareForReuse()
        disposeBag = DisposeBag()
    }

    //初始化
    override init(style: UITableViewCell.CellStyle, reuseIdentifier: String?) {
        super.init(style: style, reuseIdentifier: reuseIdentifier)
        //添加按钮
        button = UIButton(frame:CGRect(x:100, y:5, width:200, height:30))
        button.setTitle("RxSwift点击按钮", for:.normal)
        button.backgroundColor = UIColor.orange
        button.titleLabel?.font = UIFont.systemFont(ofSize: 15)
        self.contentView.addSubview(button)
    }
    
    required init?(coder aDecoder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }

    override func setSelected(_ selected: Bool, animated: Bool) {
        super.setSelected(selected, animated: animated)
        // Configure the view for the selected state
    }
}
方法三、提供给外界重用序列
extension Reactive where Base: UITableViewCell {
    // 提供给外界重用序列
    public var prepareForReuse: RxSwift.Observable<Void> {
        var prepareForReuseKey: Int8 = 0
        if let prepareForReuseOB = objc_getAssociatedObject(base, &prepareForReuseKey) as? Observable<Void> {
            return prepareForReuseOB
        }
        let prepareForReuseOB = Observable.of(
            sentMessage(#selector(Base.prepareForReuse)).map { _ in }
            , deallocated)
            .merge()
        objc_setAssociatedObject(base, &prepareForReuseKey, prepareForReuseOB, objc_AssociationPolicy.OBJC_ASSOCIATION_RETAIN)
        
        return prepareForReuseOB
    }
}
        cell.button.rx.tap.takeUntil(cell.rx.prepareForReuse)
            .subscribe(onNext: { () in
                print("点击了 \(indexPath)")
            })
方法四、提供一个重用垃圾回收袋
extension Reactive where Base: UITableViewCell {
    // 提供一个重用垃圾回收袋
    public var reuseBag: DisposeBag {
        MainScheduler.ensureExecutingOnScheduler()
        var prepareForReuseBag: Int8 = 0
        if let bag = objc_getAssociatedObject(base, &prepareForReuseBag) as? DisposeBag {
            return bag
        }

        let bag = DisposeBag()
        objc_setAssociatedObject(base, &prepareForReuseBag, bag, objc_AssociationPolicy.OBJC_ASSOCIATION_RETAIN)

        _ = sentMessage(#selector(Base.prepareForReuse))
            .subscribe(onNext: { [weak base] _ in
                let newBag = DisposeBag()
                guard let base = base else {return}
                objc_setAssociatedObject(base, &prepareForReuseBag, newBag, objc_AssociationPolicy.OBJC_ASSOCIATION_RETAIN)
            })
        return bag
    }
}
        cell.button.rx.tap.subscribe(onNext: { () in
            print("点击了 \(indexPath)")
        }).disposed(by: cell.rx.reuseBag)
方法五、扩展一个基类
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,884评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,347评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,435评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,509评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,611评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,837评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,987评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,730评论 0 267
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,194评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,525评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,664评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,334评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,944评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,764评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,997评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,389评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,554评论 2 349

推荐阅读更多精彩内容

  • 内存管理老生常谈,首先明确不是所有闭包中使用了self,就会造成循环引用,而是一定要产生循环引用链,那么在使用Rx...
    May_Dobin阅读 408评论 0 4
  • RxSwift内存管理 RxSwift 是函数响应式编程的框架。使用时满屏都是函数、闭包。那么多闭包,岂不是到处都...
    silasjs阅读 563评论 0 5
  • 本文转载自cocoachina作者:方秋枋 什么是ARC Automatic Reference Counting...
    张战威ican阅读 571评论 0 3
  • 发现 关注 消息 RxSwift入坑解读-你所需要知道的各种概念 沸沸腾关注 2016.11.27 19:11*字...
    枫叶1234阅读 2,788评论 0 2
  • 我似小草,来去无所谓。 您, 总是无私的给予我最好; 竭尽所能助我向上生长; 为我着想为我心忧; 只望我越来会越好...
    随心2018729阅读 174评论 2 1