★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★
➤微信公众号:山青咏芝(let_us_code)
➤博主域名:https://www.zengqiang.org
➤GitHub地址:https://github.com/strengthen/LeetCode
★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★
强烈推荐!!!试用博主个人App作品!提需求!提建议!
App Store搜索:【Mind Draft】
中国区可直接点击跳转:【Mind Draft】
八、变换操作(Transforming Observables)
变换操作指的是对原始的 Observable 序列进行一些转换,类似于 Swift 中 CollectionType 的各种转换。
1,buffer
(1)基本介绍
- buffer 方法作用是缓冲组合,第一个参数是缓冲时间,第二个参数是缓冲个数,第三个参数是线程。
- 该方法简单来说就是缓存 Observable 中发出的新元素,当元素达到某个数量,或者经过了特定的时间,它就会将这个元素集合发送出来。
(2)使用样例
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
|
import UIKit import RxSwift import RxCocoa class ViewController : UIViewController { let disposeBag = DisposeBag () override func viewDidLoad() { let subject = PublishSubject < String >() //每缓存3个元素则组合起来一起发出。 //如果1秒钟内不够3个也会发出(有几个发几个,一个都没有发空数组 []) subject .buffer(timeSpan: 1, count: 3, scheduler: MainScheduler .instance) .subscribe(onNext: { print ($0) }) .disposed(by: disposeBag) subject.onNext( "a" ) subject.onNext( "b" ) subject.onNext( "c" ) subject.onNext( "1" ) subject.onNext( "2" ) subject.onNext( "3" ) } } |
2,window
(1)基本介绍
- window 操作符和 buffer 十分相似。不过 buffer 是周期性的将缓存的元素集合发送出来,而 window 周期性的将元素集合以 Observable 的形态发送出来。
- 同时 buffer 要等到元素搜集完毕后,才会发出元素序列。而 window 可以实时发出元素序列。
(2)使用样例
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
|
import UIKit import RxSwift import RxCocoa class ViewController : UIViewController { let disposeBag = DisposeBag () override func viewDidLoad() { let subject = PublishSubject < String >() //每3个元素作为一个子Observable发出。 subject .window(timeSpan: 1, count: 3, scheduler: MainScheduler .instance) .subscribe(onNext: { [ weak self ] in print ( "subscribe: ($0)" ) $0.asObservable() .subscribe(onNext: { print ($0) }) .disposed(by: self !.disposeBag) }) .disposed(by: disposeBag) subject.onNext( "a" ) subject.onNext( "b" ) subject.onNext( "c" ) subject.onNext( "1" ) subject.onNext( "2" ) subject.onNext( "3" ) } } |
运行结果如下:
3,map
(1)基本介绍
- 该操作符通过传入一个函数闭包把原来的 Observable 序列转变为一个新的 Observable 序列。
(2)使用样例
1
2
3
4
5
6
|
let disposeBag = DisposeBag () Observable .of(1, 2, 3) . map { $0 * 10} .subscribe(onNext: { print ($0) }) .disposed(by: disposeBag) |
4,flatMap
(1)基本介绍
- map 在做转换的时候容易出现“升维”的情况。即转变之后,从一个序列变成了一个序列的序列。
- 而 flatMap 操作符会对源 Observable 的每一个元素应用一个转换方法,将他们转换成 Observables。 然后将这些 Observables 的元素合并之后再发送出来。即又将其 "拍扁"(降维)成一个 Observable 序列。
- 这个操作符是非常有用的。比如当 Observable 的元素本生拥有其他的 Observable 时,我们可以将所有子 Observables 的元素发送出来。
(2)使用样例
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
let disposeBag = DisposeBag () let subject1 = BehaviorSubject (value: "A" ) let subject2 = BehaviorSubject (value: "1" ) let variable = Variable (subject1) variable.asObservable() .flatMap { $0 } .subscribe(onNext: { print ($0) }) .disposed(by: disposeBag) subject1.onNext( "B" ) variable.value = subject2 subject2.onNext( "2" ) subject1.onNext( "C" ) |
5,flatMapLatest
(1)基本介绍
(2)这里我们将上例中的 flatMap 改为 flatMapLatest。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
let disposeBag = DisposeBag () let subject1 = BehaviorSubject (value: "A" ) let subject2 = BehaviorSubject (value: "1" ) let variable = Variable (subject1) variable.asObservable() .flatMapLatest { $0 } .subscribe(onNext: { print ($0) }) .disposed(by: disposeBag) subject1.onNext( "B" ) variable.value = subject2 subject2.onNext( "2" ) subject1.onNext( "C" ) |
6,flatMapFirst
(1)基本介绍
- flatMapFirst 与 flatMapLatest 正好相反:flatMapFirst 只会接收最初的 value 事件。
该操作符可以防止重复请求:
比如点击一个按钮发送一个请求,当该请求完成前,该按钮点击都不应该继续发送请求。便可该使用 flatMapFirst 操作符。
比如点击一个按钮发送一个请求,当该请求完成前,该按钮点击都不应该继续发送请求。便可该使用 flatMapFirst 操作符。
(2)这里我们将上例中的 flatMapLatest 改为 flatMapFirst。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
let disposeBag = DisposeBag () let subject1 = BehaviorSubject (value: "A" ) let subject2 = BehaviorSubject (value: "1" ) let variable = Variable (subject1) variable.asObservable() .flatMapFirst { $0 } .subscribe(onNext: { print ($0) }) .disposed(by: disposeBag) subject1.onNext( "B" ) variable.value = subject2 subject2.onNext( "2" ) subject1.onNext( "C" ) |
7,concatMap
(1)基本介绍
- concatMap 与 flatMap 的唯一区别是:当前一个 Observable 元素发送完毕后,后一个Observable 才可以开始发出元素。或者说等待前一个 Observable 产生完成事件后,才对后一个 Observable 进行订阅。
(2)这里我们将“样例3”中的 flatMap 改为 concatMap。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
let disposeBag = DisposeBag () let subject1 = BehaviorSubject (value: "A" ) let subject2 = BehaviorSubject (value: "1" ) let variable = Variable (subject1) variable.asObservable() .concatMap { $0 } .subscribe(onNext: { print ($0) }) .disposed(by: disposeBag) subject1.onNext( "B" ) variable.value = subject2 subject2.onNext( "2" ) subject1.onNext( "C" ) subject1.onCompleted() //只有前一个序列结束后,才能接收下一个序列 |
8,scan
(1)基本介绍
- scan 就是先给一个初始化的数,然后不断的拿前一个结果和最新的值进行处理操作。
(2)使用样例
1
2
3
4
5
6
7
8
|
let disposeBag = DisposeBag () Observable .of(1, 2, 3, 4, 5) .scan(0) { acum, elem in acum + elem } .subscribe(onNext: { print ($0) }) .disposed(by: disposeBag) |
9,groupBy
(1)基本介绍
- groupBy 操作符将源 Observable 分解为多个子 Observable,然后将这些子 Observable 发送出来。
- 也就是说该操作符会将元素通过某个键进行分组,然后将分组后的元素序列以 Observable 的形态发送出来。
(2)使用样例
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
let disposeBag = DisposeBag () //将奇数偶数分成两组 Observable < Int >.of(0, 1, 2, 3, 4, 5) .groupBy(keySelector: { (element) -> String in return element % 2 == 0 ? "偶数" : "基数" }) .subscribe { (event) in switch event { case .next( let group): group.asObservable().subscribe({ (event) in print ( "key:(group.key) event:(event)" ) }) .disposed(by: disposeBag) default : print ( "" ) } } .disposed(by: disposeBag) |