RxSwift

2017-08-01  本文已影响0人  花椒不麻牙

为什么要学习RxSwift

使用RxSwift可以统一的处理Delegate、KVO、Notification,可以绑定UI,方便网络请求的处理等。

RxSwift可以再单向的数据流的各个阶段都发挥作用,从而让Data的处理和流动更加的简洁和清晰。

1:通过对RxCocoa的各种回调进行统一的处理,方便了interact的处理。

2:通过对Observable的transform和composite,方便了Action的生成。

3:通过对网络请求以及其他异步数据的获取进行Observable的封装,方便了异步数据的处理。

4:通过 RxCocoa 的 binding,方便了数据的渲染。

什么是RxSwift

在说RxSwift之前,先来说下Rx,ReactiveX是一种编程模型,最初由微软开发,结合了观察者模式、迭代器模式、函数式编程的精华,来更方便地处理异步数据流。其中最重要的一个概念是Observable。

举一个简单的例子,当别人在跟你讲话时,你就是那个观察者,别人就是那个observable,它有几个特点:

1:可能会不断的跟你说话。(onNext:)

2:可能会说错话。(onError:)

3:结束会说话。(onCompleted:)

在听别人说的话后,也可以有几种反应:

1:根据说的话,做相应的事。(subscribe)

2:把对方说的话,加工下再传达给其他人。(map:)

3:参考其他人说的话再做处理。(zip:)

RxSwift  Workflow

大致分为这么几个阶段:先把Native Object变成Observable,再通过Observable内置的各种强大的转换和组合能力变成新的Observable,最后消费新的Observable的数据。

Native Object -> Observable

.rx extension

假设需要处理点击事件,正常的做法是给Tab Gesture添加一个Target-Action,然后在那里实现具体的逻辑,这样的问题在于需要重新取名字,而且丢失了上下文。RxSwift(确切的说是RxCocoa) 给系统的诸多原生控件(包括像URLSession)提供了rx扩展,所以点击的处理变成了这样:

let  tapBackground = UITapGestureRecognizer()

tapBackground.rx.event.subscribe(onNext:{[weakself]_in  

       self?.view.endEditing(true)

}).addDisposableTo(disposeBag)

view.addGestureRecognizer(tapBackground)

Observable.create

通过这个方法,可以将Native的object包装成Observable

public  func   response(_  param: Sring) -> Observable<Data>{

         return Observable.create{ observer in 

                  let task = self.dataTask(param){ (data) 

                         observer.on(.next(data))

                         observer.on(.completed)

                 }

                 task.resume()

                 return Disposables.create{

                          task.cancel()

                 }

         }

}

调用:

let disposeBag = DisposeBag()

response("ceshi").subscribe(onNext:{ data in

        print(data)

}).addDisposableTo(disposeBag)

详解:

1:Observerable返回的是一个Disposable,当执行dealloc时,会顺便执行一下Disposable.dispose(),之前创建Disposable时申请的资源就会被一并释放掉。

2:如果有多个subscriberl来subscribe(订阅)response()那么会创建多个请求,来一个observer就创建一个task,然后执行。如何让多个subscriber共享一个结果?看下文。。

Variable()

Variable(value)可以把value变成一个Observable,不过前提是使用新的赋值方式

aVariable.value = newValue

letmagicNumber=42

letmagicNumberVariable=Variable(magicNumber)

magicNumberVariable.asObservable().subscribe(onNext:{

print("magic number is\($0)")

})

magicNumberVariable.value=73

// magic number is 42

// magic number is 73

跟进去看了下,发现是通过subject来做的,大意是把value存到一个内部变量_value里,当调用value方法时,先更新_value值,然后调用内部的_subject.on(.next(newValue))方法告知 subscriber。

Subject

Subject简单来说是一个可以主动发射数据的Observable,多了onNext(value),onError(error),onCompleted()方法。

let disposeBag=DisposeBag()

let subject=PublishSubject()

subject.addObserver("1").addDisposableTo(disposeBag)

subject.onNext("🐶")

subject.onNext("🐱")

subject.addObserver("2").addDisposableTo(disposeBag)

subject.onNext("🅰️")

subject.onNext("🅱️")

在RAC时代,subject是一个不太推荐使用的功能,因为过于强大,容易失控。虽然RxSwift里没有提及太多,但还是少用为佳。

Observable -> New Observable

Observable的强大不仅在于它能实时更新value,还在于它能够被修改、过滤、组合等。这样就可以灵活的构造自己想要的数据,还不用担心数据发生变化了却不知道的情况。

combine就是把多个Observable组合起来使用,比如zip,zip对应现实的例子就是拉链,拉链需要两个元素才可以拉上去。只有当两个Observable都有了新的值时,subscribe才会被触发。

let stringSubject=PublishSubject()

let intSubject=PublishSubject()

Observable.zip(stringSubject,intSubject){stringElement,intElement in

"\(stringElement)\(intElement)"

}.subscribe(onNext:{print($0)

}).addDisposableTo(disposeBag)

stringSubject.onNext("🅰️")

stringSubject.onNext("🅱️")

intSubject.onNext(1)

intSubject.onNext(2)

// output

//// 🅰️ 1

// 🅱️ 2

如果intSubject始终没有执行onNext,那么将不会有输出,就像拉链少了一边的拉链就拉不上去了。

除了zip,还有其他的combine的方式,比如combineLatest/switchLatest等。

Transform:这是最常见的操作了,对一个Observable的数值做一些小改动,然后产出新的值,依旧是一个Observable。

let disposeBag = DisposeBag()

Observable.of(1,2,3).map{$0*$0}.subscribe(onNext:{

print($0)

}).addDisposableTo(disposeBag)

接受一个 transform 闭包,然后返回一个Observable,因为接下来使用者将会对myMap的结果进行 subscribe,所以需要在 create 内部 subscribe 一下,不然最开始的那个Observable就是个Cold Observable,一个Cold Observable是不会产生新的数据的。

Filter:是对Observable传过来的数据进行过滤,只有符合条件的才有资格被 subscribe。

Connect:在之前介绍Observable.create时有提到过,一个Observable被多次 subscribe 就会被多次触发,如果一个网络请求只想被触发一次,同时支持多个 subscriber,就可以使用publish+connect的组合。

当一个Observable使用了publish()方法后,正常的 subscribe 就不会触发它了,除非connect()方法被调用。而且每次 subscribe 不会导致Observable重新针对 observer 处理一遍。

Subscription 为什么要 Dispose?

因为有了Subscriber所以Observable被激活,然后内部就会使用各种变量来保存资源,如果不dispose的话,这些资源就会一直被 keep,很容易造成内存泄漏。

同时手动 dispose 又嫌麻烦,所以就有了DisposeBag,当这个 Bag 被回收时,Bag 里面的 subscription 会自动被 dispose,相当于从 MRC 变成了 ARC。

上一篇下一篇

猜你喜欢

热点阅读