iOS开发列表选中问题排查解决

在iOS开发中,列表View结合Rxswfit使用时,需要注意管理cell中rx变量的订阅者的生命周期,避免订阅混乱情况发生。

问题

iOS中,使用UICollectionView实现下面的颜色选择面板,当网格布局的cell数量超出一屏数量后,会复用cell,选中某个cell,会出现多个cell也被选中的情况。

背景

使用UICollectionView加载一个网格列表,并支持选中某个item,选中状态由item.isSelected控制

swift 复制代码
// UIController

        let collectionView = UICollectionView(frame: self.bounds, collectionViewLayout: layout)

        collectionView.backgroundColor = .white

        collectionView.showsVerticalScrollIndicator = false

        collectionView.bounces = false

        collectionView.contentInset = UIEdgeInsets(top: 12, left: 16, bottom: 0, right: 16)

        collectionView.register(MyCell.self, forCellWithReuseIdentifier: "cell")

  


        items.bind(to: collectionView.rx.items(cellIdentifier: "cell", cellType: MyCell.self)) { row, item, cell in

            cell.item = item

        }

        .disposed(by: self.bag)

  


  


// 这里绑定了每个cell的点击事件

        collectionView.rx.modelSelected(MyCell.self)

            .subscribe(onNext: { [weak self] item in

                guard let `self` = self else { return }

                if item.isSelected.value == true { return }

                self.foreColorItems.forEach { _item in

                    _item.isSelected.accept(false)

                }

                item.isSelected.accept(true)

                if let chooseClosure = self.chooseClosure {

                    chooseClosure(item.type, true)

                }

            })

            .disposed(by: self.bag)

  


// Item:

class Item {

    

    var type: String = ""

    var isSelected = BehaviorRelay(value: true)

    

    convenience init(type: String, isSelected: Bool) {

        self.init()

        self.type = type

        self.isSelected.accept(isSelected)

    }

}

  


  


  

Item的isSelected由BehaviorRelay管理监听,而Cell中设置item绑定到视图逻辑中,item.isSelected被声明为rxSwfit的被观察者,当isSelected值改变,会调用观察者,也就是执行 map { !$0 }.bind(),改变sImageView的isHidden。

swift 复制代码
//MyCell

  


    private var bag = DisposeBag()

    var item = CJQEditorFormatItem() {

        didSet {

            centerView.backgroundColor = UIColor(hexString: item.type.backgroundColor)

            

//            bag = DisposeBag()

            

            item.isSelected.asObservable()

                .map { !$0 }

                .bind(to: self.sImageView.rx.isHidden)

                .disposed(by: bag)

            

            sImageView.image = UIImage(named: (item.type == .foreColor_gray7 || item.type == .foreColor_gray8) ? "icon_keyboard_input_color_s_b" : "icon_keyboard_input_color_s_w")

        }

    }

分析

swift 复制代码
items.bind(to: collectionView.rx.items(cellIdentifier: "cell", cellType: MyCell.self))

{

     row, item, cell in

     cell.item = item

}

这里当复用的cell重新设置cell 的 item时,didSet中,item.isSelected又会添加一个新的观察着就是当前的cell,但是,但是item.isSelected没变,之前的cell观察者也没变,就导致了item.isSelected的观察者越来越多,改变item的isSelected,就会联动通知多个观察着cell,这里就必须要注销之前的观察者。

那么iOS 中的rxswfit 如何注销观察者。在其他语言中,可以通过显式的函数调用,来注销,但这里需要通过dispose对象的释放来注销,也就是将原来的dispose置为null,初始化一个新的。

这样就实现注销了以前的观察者,解决了点击一个cell会有多个cell响应的问题。

swift 复制代码
//MyCell

    private var bag = DisposeBag()

    var item = CJQEditorFormatItem() {

        didSet {

            centerView.backgroundColor = UIColor(hexString: item.type.backgroundColor)

  


            // 注销之前的Observer

             bag = DisposeBag()

            

            item.isSelected.asObservable()

                .map { !$0 }

                .bind(to: self.sImageView.rx.isHidden)

                .disposed(by: bag)

            

            sImageView.image = UIImage(named: (item.type == .foreColor_gray7 || item.type == .foreColor_gray8) ? "icon_keyboard_input_color_s_b" : "icon_keyboard_input_color_s_w")

        }

    }

知识点

1、UICollectionViewCell、UICollectionReusableView等视图的类或者nib文件,使用register方式可快速实现注册复用cell逻辑。

2、items.bind使用了RxSwift来管理数据源绑定。这种写法通过RxSwift的Observable和bind方法来简化数据和UI的绑定。

RxSwift的写法可以减少大量的样板代码,并且使得代码更具声明性和响应式。

3、在RxSwift中,DisposeBag 是一个用于管理和释放资源的工具。它的设计目的是为了方便地管理订阅(subscriptions)的生命周期。在DisposeBag对象引用计数为0(销毁)时,会自动的取消所有持有的订阅。

4、在cell的数据源赋值时,需要注意rx类的被订阅者的订阅管理,需要在适时通过释放disposebag来销毁所有的订阅者,重新订阅绑定到当前的item,否则会出现多个item响应的情况。

相关推荐
C澒8 分钟前
前端分层架构实战:DDD 与 Clean Architecture 在大型业务系统中的落地路径与项目实践
前端·架构·系统架构·前端框架
BestSongC12 分钟前
行人摔倒检测系统 - 前端文档(1)
前端·人工智能·目标检测
0思必得01 小时前
[Web自动化] Selenium处理滚动条
前端·爬虫·python·selenium·自动化
Misnice1 小时前
Webpack、Vite、Rsbuild区别
前端·webpack·node.js
青茶3601 小时前
php怎么实现订单接口状态轮询(二)
前端·php·接口
大橙子额2 小时前
【解决报错】Cannot assign to read only property ‘exports‘ of object ‘#<Object>‘
前端·javascript·vue.js
爱喝白开水a3 小时前
前端AI自动化测试:brower-use调研让大模型帮你做网页交互与测试
前端·人工智能·大模型·prompt·交互·agent·rag
董世昌413 小时前
深度解析ES6 Set与Map:相同点、核心差异及实战选型
前端·javascript·es6
吃杠碰小鸡4 小时前
高中数学-数列-导数证明
前端·数学·算法
kingwebo'sZone4 小时前
C#使用Aspose.Words把 word转成图片
前端·c#·word