RxTableView入门

前言

RxSwift的魅力想必用过的人都有心得体会,简直就是从入门到想放弃,从想放弃到爱不释手的过程。但是RxSwift的前世今生并不是本文想写的内容,而是其中很常用又很重要的一个部分UITableView。不管用什么语言开发移动端App,新人们基本都会被告知,掌握了TableView和CollectionView,就学会了这门语言的80%,可想而知其重要性。

RxSwift的Git社区,其中一个库RxDataSources是本文的重头戏。UITableViewDataSourceUITableViewDelegate是UITableView两个重要的代理,RxDataSources用RxSwift封装了tableView(:cellForRowAt)和tableView(:didSelectRowAt)方法,后面会细细道来。


正文

本文将从一个最最简单的TableView例子讲起,并将之用RxSwift实现,然后一步步将其功能完善。当然灵感都是来自于RxSwift官方Demo,demo中有简单的RxTableView(注:基于RxSwift的UITableView,笔者称之为RxTableView)例子。

第一个VC:Simplest UITableView

UITableViewDataSource和UITableViewDelegate是UITableView的核心,简单说明下几个最重要的代理方法:

// MARK: - UITableViewDataSource
override func numberOfSections(in tableView: UITableView) -> Int {
    return 1    // Section的数量
}
override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
    return items.count    // Section中Row的数量
}
override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
    let cell = tableView.dequeueReusableCell(withIdentifier: "reuseIdentifier", for: indexPath) // TableViewCell绘制
    cell.textLabel?.text = items[indexPath.row]
    return cell
}
// MARK: - UITableViewDelegate
override func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
    tableView.deselectRow(at: indexPath, animated: true)    // TableViewCell点击事件
    let viewController = RxTableViewController()
    viewController.type = RxTableViewType(rawValue: indexPath.row)!
    self.navigationController?.pushViewController(viewController, animated: true)
}
override func tableView(_ tableView: UITableView, heightForRowAt indexPath: IndexPath) -> CGFloat {
    return 40    // TableViewCell高度
}
第二个VC:Simplest RXTableView

RxDataSources库把UITableViewDataSource和UITableViewDelegate的代理封装成了响应式编程的风格,在代码量上少了很多,当然也更难理解:

// 1.将数据绑定到TableView上
let items = Observable.just((0..<30).map({ "\($0)"}))
items.bind(to: tableView.rx.items(cellIdentifier: "Cell", cellType: UITableViewCell.self)) { (row, element, cell) in
    cell.textLabel?.text = "row \(element)"
    }.disposed(by: disposeBag)
// 2.TableViewCell点击事件响应
tableView.rx.modelSelected(String.self).subscribe(onNext: { (item) in
    print(item)    // "1"
}).disposed(by: disposeBag)

几行代码就可以实现cell的绘制和点击处理,被观察者Obserable有一个bind(to:)方法,将自己与订阅者Observer绑定,在Obserable发送事件的时候,Observer会同步更新,两者的数据类型必须保持一致。例如Obserable<String>类型的变量str,调用bind(to:)方法将其绑定到一个UILabel类型的变量label的text属性上(同为String类型),那么str变化的同时, label.text的值跟str保持一致。

var str = "str"
let label = UILabel(frame: .zero)
Observable.of(str).bind(to: label.rx.text).disposed(by: disposeBag) // label.text = "str"
str = "changed"                                                     // label.text = "changed"

实质上,bind(to:)是对subsribe做了一层封装,subscribe(onNext:)同样可以实现上述功能:

Observable.of(str).subscribe(onNext: { label.text = $0 }).disposed(by: disposeBag)

明显,bind(to:)比subscribe(onNext:)更加"响应式"。

第三个VC:RXTableView of Sections

用items(cellIdentifier:cellType:)方法可以满足单个Section的场景,它还有另外一个方法items(dataSource:),则可以满足多个Section的场景,当然步骤也更加复杂。

// 自定义Model,遵循SectionModelType
struct SectionModel<HeaderType, ItemType>: SectionModelType {
    var header: HeaderType
    var items: [ItemType]
    
    init(header: HeaderType, items: [ItemType]) {
        self.header = header
        self.items = items
    }
    
    init(original: SectionModel<HeaderType, ItemType>, items: [ItemType]) {
        self.header = original.header
        self.items = items
    }
}
// 1.创建DataSource
let dataSource = RxTableViewSectionedReloadDataSource<SectionModel<String,Int>>(configureCell: { (section, tableView, indexPath, element) in
    let cell = tableView.dequeueReusableCell(withIdentifier: "Cell")!
    cell.textLabel?.text = "row \(element)"
    return cell
})
// 2.设置HeaderTitle(可?。?dataSource.titleForHeaderInSection = { (dataSource, sectionIndex) -> String? in
    return dataSource[sectionIndex].header
}
// 3.将数据绑定到TableView上
let items = [SectionModel(header: "section 1", items: [1, 2, 3]), SectionModel(header: "section 2", items: [1, 2, 3])]
Observable.just(items).bind(to: tableView.rx.items(dataSource: dataSource)).disposed(by: disposeBag)
// 4.TableViewCell点击事件响应
tableView.rx.itemSelected.map {
    return ($0, items[$0.section].header, dataSource[$0])
}.subscribe(onNext: { indexPath, header, item in
    print("\(header), row \(item)")    // "section 1, row 1"
}).disposed(by: disposeBag)

相对于上一个VC的直接绑定,这里需要额外定义Model,创建DataSource,并调用tableView.rx.items(dataSource:)方法来进行绑定,这些额外的步骤都是为了提高可定制化的程度。这里Cell的点击事件响应已经由modelSelected()方法换成了itemSelected()方法,差别就是modelSelected()方法会直接返回选中的model,而不会返回选中model的indexPath。


RxDataSources的优劣

RxDataSources库已经提供了强大的功能来满足RxSwift开发者们实现响应式TableView的迫切愿望,但是也有不完美的地方,笔者将结合自身经历列出优势和不足:

优势
  1. 与RxSwift结合,让代码更加“响应式”。即使用复写delegate方法来实现UITableViewDataSource和UITableViewDelegate也是完全OK的,只是面向对象编程与响应式编程的代码糅合在一起,显示特别奇怪。一切皆“响应”,没有什么是响应式编程无法实现的,特别不能用面向对象的思维去思考,会越走越远。
  2. 类似闭包语法的风格,让代码更加简洁。以前笔者喜欢用delegate,觉得逻辑很简单直观,但是久了就会发现调试起来,页面上下滚动太频繁,对象的创建和代理方法调用基本不在同一页面。RxDataSources这种“闭包式”的语法,让代码变得更加简洁,更易调试。
  3. 其他优点应该还有很多。。。。。
不足

尚未完全Rx化,不得不复写代理方法。笔者用了一段时间这个库,目前发现有两点是无法做到响应式实现

  1. 通常在调用tableView(:didSelectRowAt:)方法后,会调用deselectRow(at:animated:) 函数来取消选中,但是目前没有发现调用modelSelected()或itemSelected()方法后有什么方法可以直接取消选中,而不是在subscribe(onNext:)方法中用面向对象的逻辑实现;
  2. tableView(: heightForRowAt: )方法是用来设置高度的,RxTableViewSectionedReloadDataSource的配置项中目前没有发现有什么方法可以设置Cell高度。

以上都是个人观点,假如是因为笔者才疏学浅才没有发现的话,还望指正。


总结

RxTableView的入门讲解就告一段落,后续会出一篇进阶篇,深入分析数据和视图的绑定机制,并实现一(数据)对多(视图)的绑定,一(数据)对一(多视图切换)的绑定,也是自己在实际项目中遇到的坑,拿出来跟各位共同探讨。

Demo地址:https://github.com/MrSuperJJ/RxTableViewDemo

Git地址:https://github.com/MrSuperJJ

简书地址:http://08643.cn/u/9079457bd1de

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

推荐阅读更多精彩内容

  • 1.ios高性能编程 (1).内层 最小的内层平均值和峰值(2).耗电量 高效的算法和数据结构(3).初始化时...
    欧辰_OSR阅读 29,353评论 8 265
  • 概述在iOS开发中UITableView可以说是使用最广泛的控件,我们平时使用的软件中到处都可以看到它的影子,类似...
    liudhkk阅读 9,021评论 3 38
  • 奈良第二日,照旧晚起,照旧乱逛。租了两辆自行车,漫无目的地骑在冬日暖阳里。骑过正在修复的平城宫,骑过数条铁轨,骑过...
    诺拉的以后阅读 204评论 0 1
  • 前几天一桌媒体吃饭,大家聊起了米锤之类的手机,不免要问我为什么那么看,我跟他们说完了,他们都表示非常认同我的观点。...
    孙见阳阅读 279评论 0 1
  • 看了天气预报 海口15日刮台风 于是马不停蹄的赶往了三亚 一路上的狂风暴雨 这天气预报怎么跟翻书似的 一点也不靠谱...
    林恩恩恩阅读 111评论 0 0