RxSwift 入门以及了解

前言介绍

RX是一个帮助我们简化异步编程的框架。它拓展了观察者模式,使我们可以自由组合多个异步事件,而不需要去关心线程,同步,线程安全,并发数据以及I/O阻塞。RXSwift是RX的Swift版本。

为什么要使用RxSwift

  • 复合
  • 复用
  • 清晰
  • 易用 - 因为它抽象了异步编程,使我们统一了代码风格
  • 稳定 - 因为Rx使完全通过单元测试的
Target Action
button.rx.tap.subscribe(onNext: {
    print("button Tapped")
})
.disposed(by: disposeBag)
代理
scrollView.rx.contentOffset
            .subscribe(onNext: { contentOffset in
                print("contentOffset: \(contentOffset)")
            })
            .disposed(by: disposeBag)
闭包回调
URLSession.shared.rx.data(request: URLRequest(url: url))
    .subscribe(onNext: { data in
        print("Data Task Success with count: \(data.count)")
    }, onError: { error in
        print("Data Task Error: \(error)")
    })
    .disposed(by: disposeBag)
通知
NotificationCenter.default.rx
        .notification(.UIApplicationWillEnterForeground)
        .subscribe(onNext: { (notification) in
            print("Application Will Enter Foreground")
        })
        .disposed(by: disposeBag)

不需要去管理观察者的生命周期,这样你就有更多精力去关注业务逻辑。

KVO
user.rx.observe(String.self, #keyPath(User.name))
        .subscribe(onNext: { newValue in
            print("do something with newValue")
        })
        .disposed(by: disposeBag)
多个任务之间有依赖关系
/// 用 Rx 封装接口
enum Api {

    /// 通过用户名密码取得一个 token
    static func token(username: String, password: String) -> Observable<String> { ... }

    /// 通过 token 取得用户信息
    static func userInfo(token: String) -> Observable<UserInfo> { ... }
}
/// 通过用户名和密码获取用户信息
Api.token(username: "beeth0ven", password: "987654321")
    .flatMapLatest(Api.userInfo)
    .subscribe(onNext: { userInfo in
        print("获取用户信息成功: \(userInfo)")
    }, onError: { error in
        print("获取用户信息失败: \(error)")
    })
    .disposed(by: disposeBag)
等待多个并发任务完成后处理结果
/// 用 Rx 封装接口
enum Api {

    /// 取得老师的详细信息
    static func teacher(teacherId: Int) -> Observable<Teacher> { ... }

    /// 取得老师的评论
    static func teacherComments(teacherId: Int) -> Observable<[Comment]> { ... }
}
/// 同时取得老师信息和老师评论
Observable.zip(
      Api.teacher(teacherId: teacherId),
      Api.teacherComments(teacherId: teacherId)
    ).subscribe(onNext: { (teacher, comments) in
        print("获取老师信息成功: \(teacher)")
        print("获取老师评论成功: \(comments.count) 条")
    }, onError: { error in
        print("获取老师信息或评论失败: \(error)")
    })
    .disposed(by: disposeBag)

函数响应式编程

[图片上传失败...(image-6266d-1598506756093)]

Observable 理解成Signal,订阅Signal去完成一些处理操作。比如绑定数据,最后销毁这个Signal。

函数式编程是种编程范式,它需要我们将函数作为参数传递,或者作为返回值返还。我们可以通过组合不同的函数来得到想要的结果。

对每个事件都做出一些响应,所以这种编程叫做函数响应式编程。我们通过不同的构建函数,来创建所需要的数据序列。最后通过适当的方式来响应这个序列。这就是函数响应式编程。

数据绑定

数据绑定(订阅):就是指将可被监听的序列绑定到观察者上。

比如:

let image: Observable<UIImage> = ...
image.bind(to: imageView.rx.image)

将一个图片序列 “同步” 到imageView上。这个序列里面的图片可以是异步产生的。这里定义的 image 就是上图中蓝色部分(可被监听的序列)。这种“同步机制”就是数据绑定(订阅)。

RXSwift核心

[图片上传失败...(image-9ce6ac-1598506756094)]

  • Observable - 产生事件(即信号)
  • Observer - 响应事件(即绑定信号后的处理)
  • Operator - 创建变化组合事件(组合多个信号,并行或者串行)
  • Disposable - 管理绑定(订阅)的生命周期(一般管理信号销毁居多)
  • Schedules - 线程队列调配
// Observable<String>
let text = usernameOutlet.rx.text.orEmpty.asObservable()

// Observable<Bool>
let passwordValid = text
    // Operator
    .map { $0.characters.count >= minimalUsernameLength }

// Observer<Bool>
let observer = passwordValidOutlet.rx.isHidden

// Disposable
let disposable = passwordValid
    // Scheduler 用于控制任务在那个线程队列运行
    .subscribeOn(MainScheduler.instance)
    .observeOn(MainScheduler.instance)
    .bind(to: observer)


...

// 取消绑定,你可以在退出页面时取消绑定
disposable.dispose()

Observable - 可被监听的序列

所有的事物都是序列,是信号。它能异步的触发一系列事件并携带不可更改的状态变量。简单来说,它能让某个类的实例在一段事件内实现对另一个实例对象值的观察。例如:观察者可以捕获对所有可观察对象触发的事件,从而实现UI的实时更新或者是数据的实时处理。

subscribe后面的onNext,onError, onCompleted 分别响应。这些为事件。completed事件和error事件都会终止可观察对象的生命周期。不会继续触发新的事件。

public enum Event<Element> {
    case next(Element)
    case error(Swift.Error)
    case completed
}
  • next - 序列产生了一个新的元素
  • error - 创建序列时产生了一个错误,导致序列终止
  • completed - 序列的所有元素都已经成功产生,整个序列已经完成
Single
  • 要么产生一个元素,要么产生一个error事件
  • 不会共享状态变化
Completable
  • 发出零个元素
  • 发出一个completed事件或者一个error事件
  • 不会共享状态变化
Maybe
  • 发出一个元素或者一个completed事件或者一个error事件
  • 不会共享状态变化
Driver
  • 不会产生error事件
  • 一定在MainScheduler监听(主线程监听)
  • 共享状态变化
ControlEvent

专门用于描述UI控件所产生的事件。

  • 不会产生error事件
  • 一定在MainScheduler订阅(主线程订阅)
  • 一定在MainScheduler监听(主线程监听)
  • 共享状态变化

Sequences - 序列

有限观察序列 (Finite observable sequences)

该序列是指那些最后会以completed或者error事件终止生命周期的可观察对象。
比如网络任务:

API.download(file: "http://www...")
    .subscribe( onNext: { data in
                        append data to temporary file }, 
                onError: { error in
                        display error to user }, 
                onCompleted: {
                        use downloaded file })
无限观察序列 (Infinite observable sequences)

与网络任务不同的是,UI以及交互事件都是无限观察序列。它们并不存在一个明确的生命周期终止。
比如:

UIDevice.rx.orientation.subscribe(onNext: { current in 
    switch current { 
        case .landscape:
            re-arrange UI for landscape 
        case .portrait:
            re-arrange UI for portrait 
    } 
})

Observer - 观察者

观察者是用来监听事件,然后它需要这个事件做出响应。

AnyObserver

可以用来描述任意一种观察者

Binder
  • 不会处理错误事件
  • 确保绑定都是在给定Scheduler上执行(默认MainScheduler

一旦产出错误事件,在调试环境下将执行fatalError,在发布环境下将打印错误信息。

Operator - 操作符

操作符可以帮助我们创建新的序列,或者变化组合原有的序列,从而生成一个新的序列。

filter - 过滤
map - 转换
zip - 配对

还有很多,这里暂时不展开太多。

Disposable - 可被清除的资源

通常来说,一个序列如果发出了error或者completed事件,那么所有内部资源都会被释放。如果需要提前释放或取消订阅,那么可以使用dispose方法。

一般我们推荐使用清除包(DisposeBag)或者takeUntil操作符来管理订阅的生命周期。

DisposeBag - 清除包
var disposeBag = DisposeBag()

override func viewWillAppear(_ animated: Bool) {
    super.viewWillAppear(animated)

    textField.rx.text.orEmpty
        .subscribe(onNext: { text in print(text) })
        .disposed(by: self.disposeBag)
}

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

    self.disposeBag = DisposeBag()
}

清除包被释放的时候,清除包内部所有可被清除的资源(Disposable)都将被清除。

takeUntil
override func viewDidLoad() {
    super.viewDidLoad()

    ...

    _ = usernameValid
        .takeUntil(self.rx.deallocated)
        .bind(to: passwordOutlet.rx.isEnabled)

    _ = usernameValid
        .takeUntil(self.rx.deallocated)
        .bind(to: usernameValidOutlet.rx.isHidden)

    _ = passwordValid
        .takeUntil(self.rx.deallocated)
        .bind(to: passwordValidOutlet.rx.isHidden)

    _ = everythingValid
        .takeUntil(self.rx.deallocated)
        .bind(to: doSomethingOutlet.rx.isEnabled)

    _ = doSomethingOutlet.rx.tap
        .takeUntil(self.rx.deallocated)
        .subscribe(onNext: { [weak self] in self?.showAlert() })
}

这将使得订阅一直持续到控制器的dealloc事件产生为止。

Schedulers - 调度器

SchedulesRX实现多线程的核心模块,主要用于控制任务在哪个线程或队列运行。

在线程这部分主要有两个操作符:observeOn 和 subscribeOn ,常用的还是 observeOn 。

调用 observeOn 指定接下来的操作在哪个线程。
调用 subscribeOn 决定订阅者的操作执行在哪个线程。

let rxData: Observable<Data> = ...

rxData
    .subscribeOn(ConcurrentDispatchQueueScheduler(qos: .userInitiated))
    .observeOn(MainScheduler.instance)
    .subscribe(onNext: { [weak self] data in
        self?.data = data
    })
    .disposed(by: disposeBag)
使用subscribeOn

subscribeOn来决定数据序列的构建函数在哪个Scheduler

使用observeOn

observeOn来决定在哪个Scheduler监听这个数据序列.

一个比较典型的例子就是,在后台发起网络请求,然后解析数据,最后在主线程刷新页面。你就可以先用 subscribeOn 切到后台去发送请求并解析数据,最后用 observeOn 切换到主线程更新页面。

MainScheduler

主线程

SerialDispatchQueueScheduler

串行队列

ConcurrentDispatchQueueScheduler

并行队列

OperationQueueScheduler

可以理解成NSOperationQueue.

Error Handling - 错误处理

一旦序列里面发生了一个error事件,整个序列将被终止。

  • retry - 重试
  • catch - 恢复
retry
retryWhen

延时后重试

catchError

catchError可以在错误产生时,用一个备用元素或者一组备用元素将错误替换掉。

result

可以将错误转换成result,即使发生了错误,序列也不会终止。

Subject

Subject是一个桥梁,既是 Observable 也是 Observer

  • 作为一个 Observer,它可以订阅序列
  • 同时作为一个Obserable,它可以转发或者发射数据。

Subject有几种:

  • PublishSubject
  • ReplaySubject
  • BehaviorSubject
  • Variable
PublishSubject

当有观察者订阅PublishSubject时,PublishSubject会发射订阅之后的数据给这个观察者。(存在数据丢失的问题)

ReplaySubject

和PublishSubject不同,不论观察者什么时候订阅,ReplaySubject都会发射完整的数据给观察者。

BehaviorSubject

当一个观察者订阅一个BehaviorSubject,它会发送原序列最近的那个值(如果还没有值会有默认的),之后继续发射原序列的值。

Variable

Variable是BehaviorSubject的一个封装,但是不会因为错误终止也不会正常终止,是一个无限序列。

注意的问题

cell中使用button重复订阅的问题

解决办法:为Cell添加一个DisposeBag,在prepareForReuse时,更换新的DisposeBag

var disposeBag = DisposeBag()

override func prepareForReuse() {
    super.prepareForReuse()
    disposeBag = DisposeBag()
}

冷、热信号的问题

热信号,一般是我们接触到的最新的东西。
冷信号,一般是依赖其他的信号订阅后的行为,比如一些异步操作。

耦合性问题

RXSwift滥用会导致项目耦合性变差,可读性和可维护性都会变得很差,并且有一定的学习成本,会给项目本身带来一定的风险。

循环引用问题

RXSwift使用过程中会大量使用block,使用不当容易导致循环引用。

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