1. 组合操作符
1.1 startWith
print("*****startWith*****")
Observable.of("1", "2", "3", "4")
.startWith("A")
.startWith("B")
.startWith("C", "a", "b")
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
//效果: CabBA1234
1.2 merge
- 将源可观察序列中的元素组合成一个新的可观察序列,并将像每个源可观察序列发出元素一样发出每个元素
print("*****merge*****")
let subject1 = PublishSubject()
let subject2 = PublishSubject()
// merge subject1和subject2
Observable.of(subject1, subject2)
.merge()
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
subject1.onNext("C")
subject1.onNext("o")
subject2.onNext("o")
subject2.onNext("o")
subject1.onNext("c")
subject2.onNext("i")
//任何一个响应都会勾起新序列响应
1.3 zip
- 将多达8个源可观测序列组合成一个新的可观测序列,并将从组合的可观测序列中发射出对应索引处每个源可观测序列的元素
print("*****zip*****")
let stringSubject = PublishSubject()
let intSubject = PublishSubject()
Observable.zip(stringSubject, intSubject) { stringElement, intElement in
"\(stringElement) \(intElement)"
}
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
stringSubject.onNext("C")
stringSubject.onNext("o") // 到这里存储了 C o 但是不会响应除非;另一个响应
intSubject.onNext(1) // 勾出一个
intSubject.onNext(2) // 勾出另一个
stringSubject.onNext("i") // 存一个
intSubject.onNext(3) // 勾出一个
// 说白了: 只有两个序列同时有值的时候才会响应,否则存值
1.4 combineLatest
- 将8源可观测序列组合成一个新的观测序列,并将开始发出联合观测序列的每个源的最新元素可观测序列一旦所有排放源序列至少有一个元素,并且当源可观测序列发出的任何一个新元素
print("*****combineLatest*****")
let stringSub = PublishSubject()
let intSub = PublishSubject()
Observable.combineLatest(stringSub, intSub) { strElement, intElement in
"\(strElement) \(intElement)"
}
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
stringSub.onNext("L") // 存一个 L
stringSub.onNext("G") // 存了一个覆盖 - 和zip不一样
intSub.onNext(1) // 发现strOB也有G 响应 G 1
intSub.onNext(2) // 覆盖1 -> 2 发现strOB 有值G 响应 G 2
stringSub.onNext("AAAAAA") // 覆盖G -> AAAAAA 发现intOB 有值2 响应 AAAAAA 2
// combineLatest 比较zip 会覆盖
// 应用非常频繁: 比如账户和密码同时满足->才能登陆. 不关系账户密码怎么变化的只要查看最后有值就可以 loginEnable
1.5 switchLatest
- 将可观察序列发出的元素转换为可观察序列,并从最近的内部可观察序列发出元素
print("*****switchLatest*****")
let switchLatestSub1 = BehaviorSubject(value: "L")
let switchLatestSub2 = BehaviorSubject(value: "1")
let switchLatestSub = BehaviorSubject(value: switchLatestSub1)// 选择了 switchLatestSub1 就不会监听 switchLatestSub2
switchLatestSub.asObservable()
.switchLatest()
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
switchLatestSub1.onNext("G")
switchLatestSub1.onNext("_")
switchLatestSub2.onNext("2")
switchLatestSub2.onNext("3") // 2-3都会不会监听,但是默认保存由 2覆盖1 3覆盖2
switchLatestSub.onNext(switchLatestSub2) // 切换到 switchLatestSub2
switchLatestSub1.onNext("*")
switchLatestSub1.onNext("BBBBBB") // 原理同上面 下面如果再次切换到 switchLatestSub1会打印出 BBBBBB
switchLatestSub2.onNext("4")
2. 映射操作符
2.1 map
- 转换闭包应用于可观察序列发出的元素,并返回转换后的元素的新可观察序列。
print("*****map*****")
let ob = Observable.of(1,2,3,4)
ob.map { (number) -> Int in
return number+2
}
.subscribe{
print("\($0)")
}
.disposed(by: disposeBag)
2.2 flatMap: and flatMapLatest:
- 将可观测序列发射的元素转换为可观测序列,并将两个可观测序列的发射合并为一个可观测序列。
- 这也很有用,例如,当你有一个可观察的序列,它本身发出可观察的序列,你想能够对任何一个可观察序列的新发射做出反应(序列中序列:比如网络序列中还有模型序列)
-
latMap
和flatMapLatest
的区别是,flatMapLatest
只会从最近的内部可观测序列发射元素
print("*****flatMap*****")
let boy = LGPlayer(score: 100)
let girl = LGPlayer(score: 90)
let player = BehaviorSubject(value: boy)
player.asObservable()
.flatMap { $0.score.asObservable() } // 本身score就是序列 模型就是序列中的序列
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
boy.score.onNext(60)
player.onNext(girl)
boy.score.onNext(50)
boy.score.onNext(40)// 如果切换到 flatMapLatest 就不会打印
girl.score.onNext(10)
girl.score.onNext(0)
// flatMapLatest实际上是map和switchLatest操作符的组合。
2.3 scan
- 从初始就带有一个默认值开始,然后对可观察序列发出的每个元素应用累加器闭包,并以单个元素可观察序列的形式返回每个中间结果
print("*****scan*****")
Observable.of(10, 100, 1000)
.scan(2) { aggregateValue, newValue in
aggregateValue + newValue // 10 + 2 , 100 + 10 + 2 , 1000 + 100 + 2
}
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
// 这里主要强调序列值之间的关系
3. 过滤条件操作符
3.1 filter
print("*****filter*****")
Observable.of(1,2,3,4,5,6,7,8,9,0)
.filter { $0 % 2 == 0 }
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
3.2 distinctUntilChanged
print("*****distinctUntilChanged*****")
Observable.of("1", "2", "2", "2", "3", "3", "4")
.distinctUntilChanged()
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
3.3 elementAt
print("*****elementAt*****")
Observable.of("A", "B", "C", "D", "E")
.elementAt(3)
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
3.4 single
- 只发出可观察序列发出的第一个元素(或满足条件的第一个元素)。如果可观察序列发出多个元素,将抛出一个错误。
print("*****single*****")
Observable.of("AAA", "BBB")
.single()
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
Observable.of("AAA", "BBB")
.single { $0 == "BBB" }
.subscribe { print($0) }
.disposed(by: disposeBag)
3.5 take
- 只从一个可观察序列的开始发出指定数量的元素。 上面
signal
只有一个序列 在实际开发会受到局限 这里引出 take
想几个就几个
print("*****take*****")
Observable.of("AAA", "BBB","CCC", "DDD")
.take(2)
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
3.6 takeLast
print("*****takeLast*****")
Observable.of("AAA", "BBB","CCC", "DDD")
.takeLast(3)
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
3.7 takeWhile
- 只要指定条件的值为
true
,就从可观察序列的开始发出元素
Observable.of(1, 2, 3, 4, 5, 6)
.takeWhile { $0 < 3 }
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
3.8 takeUntil
- 从源可观察序列发出元素,直到参考可观察序列发出元素
- 这个要重点,应用非常频繁 比如我页面销毁了,就不能获取值了(cell重用运用)
print("*****takeUntil*****")
let sourceSequence = PublishSubject()
let referenceSequence = PublishSubject()
sourceSequence
.takeUntil(referenceSequence)
.subscribe { print($0) }
.disposed(by: disposeBag)
sourceSequence.onNext("AAA")
sourceSequence.onNext("BBB")
sourceSequence.onNext("CCC")
referenceSequence.onNext("BBB") // 条件一出来,下面就走不了
sourceSequence.onNext("DDD")
sourceSequence.onNext("EEE")
sourceSequence.onNext("FFF")
3.9 skip
- 从源可观察序列发出元素,直到参考可观察序列发出元素
- 这个要重点,应用非常频繁 不用解释
textfiled
都会有默认序列产生
print("*****skip*****")
Observable.of(1, 2, 3, 4, 5, 6)
.skip(2)
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
print("*****skipWhile*****")
Observable.of(1, 2, 3, 4, 5, 6)
.skipWhile { $0 < 4 }
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
3.10 skipUntil
- 抑制从源可观察序列发出元素,直到参考可观察序列发出元素
print("*****skipUntil*****")
let sourceSeq = PublishSubject()
let referenceSeq = PublishSubject()
sourceSeq
.skipUntil(referenceSeq)
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
// 没有条件命令 下面走不了
sourceSeq.onNext("AAA")
sourceSeq.onNext("BBB")
sourceSeq.onNext("CCC")
referenceSeq.onNext("BBB") // 条件一出来,下面就可以走了
sourceSeq.onNext("DDD")
sourceSeq.onNext("EEE")
sourceSeq.onNext("FFF")