组合操作符
一、startWith
在开始从可观察源发出元素之前,发出指定的元素序列
Observable.of("1", "2")
.startWith("A")
.startWith("C", "a")
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
/*打印结果:
C
a
A
1
2
/*
二、merge
将源可观察序列中的元素组合成一个新的可观察序列,并将像每个源可观察序列发出元素一样发出每个元素
let subject1 = PublishSubject<String>()
let subject2 = PublishSubject<String>()
// merge subject1和subject2
Observable.of(subject1, subject2)
.merge()
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
subject1.onNext("J")
subject1.onNext("e")
subject2.onNext("n")
subject2.onNext("s")
subject1.onNext("e")
subject2.onNext("n")
/*打印结果:
J
e
n
s
e
n
任何一个响应都会勾起新序列响应*/
三、zip
将多达8个源可观测序列组合成一个新的可观测序列,并将从组合的可观测序列中发射出对应索引处每个源可观测序列的元素
Observable.zip(stringSubject, intSubject) { stringElement, intElement in
"\(stringElement) \(intElement)"
}
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
stringSubject.onNext("J")
stringSubject.onNext("E")
intSubject.onNext(1) // 勾出一个
intSubject.onNext(4) // 勾出另一个
intSubject.onNext(3) // 勾出一个
stringSubject.onNext("N") // 存一个
/*打印结果:
J 1
E 4
N 3
只有两个序列同时有值的时候才会响应,否则存值
*/
四、combineLatest
将可观测序列组合成一个新的观测序列,并将开始发出联合观测序列的每个源的最新元素可观测序列一旦所有排放源序列至少有一个元素,并且当源可观测序列发出的任何一个新元素
let stringSub = PublishSubject<String>()
let intSub = PublishSubject<Int>()
Observable.combineLatest(stringSub, intSub) { strElement, intElement in
"\(strElement) \(intElement)"
}
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
stringSub.onNext("J") // 存一个 J
intSub.onNext(1) // 发现stringSub有J 响应 J 1
intSub.onNext(2) // 覆盖2 -> 1 发现stringSub 有值J 响应 J 2
stringSub.onNext("E") // E 2
stringSub.onNext("JENSEN")
/*打印结果:
J 1
J 2
E 2
JENSEN 2
只有两个序列同时有值的时候才会响应,否则存值
应用非常频繁: 比如账户和密码同时满足->才能登陆. 不关系账户密码怎么变化的只要查看最后有值就可以 loginEnable
*/
五、switchLatest
将可观察序列发出的元素转换为可观察序列,并从最近的内部可观察序列发出元素
let switchLatestSub1 = BehaviorSubject(value: "L")
let switchLatestSub2 = BehaviorSubject(value: "1")
let switchLatestSub = BehaviorSubject(value: switchLatestSub2)// 选择了 switchLatestSub1 就不会监听 switchLatestSub2
switchLatestSub.asObservable()
.switchLatest()
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
switchLatestSub1.onNext("G")
switchLatestSub1.onNext("_")
switchLatestSub2.onNext("2")
switchLatestSub2.onNext("3")
switchLatestSub.onNext(switchLatestSub2)
switchLatestSub1.onNext("*")
switchLatestSub1.onNext("JENSEN")
switchLatestSub2.onNext("4")
映射操作符
一、map
转换闭包应用于可观察序列发出的元素,并返回转换后的元素的新可观察序列。
let ob = Observable.of(1,2,3,4)
ob.map { (number) -> Int in
return number+2
}
.subscribe{
print("\($0)")
}
.disposed(by: disposeBag)
二、flatMap and flatMapLatest
将可观测序列发射的元素转换为可观测序列,并将两个可观测序列的发射合并为一个可观测序列。
例如,当你有一个可观察的序列,它本身发出可观察的序列,你想能够对任何一个可观察序列的新发射做出反应(序列中序列:比如网络序列中还有模型序列)
flatMap和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) 如果切换到 flatMapLatest 就不会打印
boy.score.onNext(40)// 如果切换到 flatMapLatest 就不会打印
girl.score.onNext(10)
girl.score.onNext(0)
// flatMapLatest实际上是map和switchLatest操作符的组合。
三、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)
// 这里主要强调序列值之间的关系
过滤条件操作符
一、filter
仅从满足指定条件的可观察序列中发出那些元素
Observable.of(1,2,3,4,5,6,7,8,9,0)
.filter { $0 % 2 == 0 }
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
二、distinctUntilChanged
抑制可观察序列发出的顺序重复元素
Observable.of("1", "2", "2", "2", "3", "3", "4")
.distinctUntilChanged()
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
三、elementAt
仅在可观察序列发出的所有元素的指定索引处发出元素
Observable.of("J", "e", "n", "s", "e")
.elementAt(3)
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
四、single
只发出可观察序列发出的第一个元素(或满足条件的第一个元素)。如果可观察序列发出多个元素,将抛出一个错误。
Observable.of("888", "999")
.single()
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
Observable.of("888", "999")
.single { $0 == "999" }
.subscribe { print($0) }
.disposed(by: disposeBag)
五、take
只从一个可观察序列的开始发出指定数量的元素。 上面signal只有一个序列 在实际开发会受到局限 这里引出 take 想几个就几个
Observable.of("111", "222","333", "444")
.take(2)
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
六、takeLast
仅从可观察序列的末尾发出指定数量的元素
Observable.of("111", "222","333", "444")
.takeLast(3)
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
七、takeWhile
只要指定条件的值为true,就从可观察序列的开始发出元素
Observable.of(1, 2, 3, 4, 5, 6)
.takeWhile { $0 < 3 }
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
八、takeUntil
从源可观察序列发出元素,直到参考可观察序列发出元素
// 这个要重点,应用非常频繁 比如我页面销毁了,就不能获取值了(cell重用运用)
let sourceSequence = PublishSubject<String>()
let referenceSequence = PublishSubject<String>()
sourceSequence
.takeUntil(referenceSequence)
.subscribe { print($0) }
.disposed(by: disposeBag)
sourceSequence.onNext("J")
sourceSequence.onNext("E")
sourceSequence.onNext("N")
referenceSequence.onNext("S") // 条件一出来,下面就走不了
sourceSequence.onNext("E")
sourceSequence.onNext("N")
九、skip
从源可观察序列发出元素,直到参考可观察序列发出元素
// 这个要重点,应用非常频繁 不用解释 textfiled 都会有默认序列产生
Observable.of(1, 2, 3, 4, 5, 6)
.skip(2)
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
十、skipWhile
Observable.of(1, 2, 3, 4, 5, 6)
.skipWhile { 0) })
.disposed(by: disposeBag)
十一、skipUntil
抑制从源可观察序列发出元素,直到参考可观察序列发出元素
let sourceSeq = PublishSubject<String>()
let referenceSeq = PublishSubject<String>()
sourceSeq
.skipUntil(referenceSeq)
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
// 没有条件命令 下面走不了
sourceSeq.onNext("J")
sourceSeq.onNext("E")
sourceSeq.onNext("N")
referenceSeq.onNext("S") // 条件一出来,下面就可以走了
sourceSeq.onNext("E")
sourceSeq.onNext("N")
集合控制操作符
一、toArray
将一个可观察序列转换为一个数组,将该数组作为一个新的单元素可观察序列发出,然后终止
Observable.range(start: 1, count: 10)
.toArray()
.subscribe { print($0) }
.disposed(by: disposeBag)
二、reduce
从一个设置的初始化值开始,然后对一个可观察序列发出的所有元素应用累加器闭包,并以单个元素可观察序列的形式返回聚合结果 - 类似scan
Observable.of(10, 100, 1000)
.reduce(1, accumulator: +) // 1 + 10 + 100 + 1000 = 1111
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
三、concat
以顺序方式连接来自一个可观察序列的内部可观察序列的元素,在从下一个序列发出元素之前,等待每个序列成功终止
let subject1 = BehaviorSubject(value: "Hank")
let subject2 = BehaviorSubject(value: "1")
let subjectsSubject = BehaviorSubject(value: subject1)
subjectsSubject.asObservable()
.concat()
.subscribe { print($0) }
.disposed(by: disposeBag)
subject1.onNext("Cooci")
subject1.onNext("Kody")
subjectsSubject.onNext(subject2)
subject2.onNext("打印不出来")
subject2.onNext("2")
subject1.onCompleted() // 必须要等subject1 完成了才能订阅到! 用来控制顺序 网络数据的异步
subject2.onNext("3")
subject2.onNext("4")
从可观察对象的错误通知中恢复的操作符
一、catchErrorJustReturn
从错误事件中恢复,方法是返回一个可观察到的序列,该序列发出单个元素,然后终止
let sequenceThatFails = PublishSubject<String>()
sequenceThatFails
.catchErrorJustReturn("J")
.subscribe { print($0) }
.disposed(by: disposeBag)
sequenceThatFails.onNext("E")
sequenceThatFails.onNext("N") // 正常序列发送成功的
sequenceThatFails.onError(self.lgError) //发送失败的序列,一旦订阅到位 返回我们之前设定的错误的预案
二、catchError
通过切换到提供的恢复可观察序列,从错误事件中恢复
let recoverySequence = PublishSubject<String>()
sequenceThatFails
.catchError {
print("Error:", $0)
return recoverySequence // 获取到了错误序列-我们在中间的闭包操作处理完毕,返回给用户需要的序列(showAlert)
}
.subscribe { print($0) }
.disposed(by: disposeBag)
sequenceThatFails.onNext("J")
sequenceThatFails.onNext("E") // 正常序列发送成功的
sequenceThatFails.onError(lgError) // 发送失败的序列
recoverySequence.onNext("N")
三、retry
通过无限地重新订阅可观察序列来恢复重复的错误事件
var count = 1 // 外界变量控制流程
let sequenceRetryErrors = Observable<String>.create { observer in
observer.onNext("J")
observer.onNext("E")
observer.onNext("N")
if count == 1 { // 流程进来之后就会过度-这里的条件可以作为出口,失败的次数
observer.onError(self.lgError) // 接收到了错误序列,重试序列发生
print("错误序列来了")
count += 1
}
observer.onNext("S")
observer.onNext("E")
observer.onNext("N")
observer.onCompleted()
return Disposables.create()
}
sequenceRetryErrors
.retry()
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
四、retry(_:)
通过重新订阅可观察到的序列,重复地从错误事件中恢复,直到重试次数达到max未遂计数
let sequenceThatErrors = Observable<String>.create { observer in
observer.onNext("J")
observer.onNext("E")
observer.onNext("N")
if count < 5 { // 这里设置的错误出口是没有太多意义的额,因为我们设置重试次数
observer.onError(self.lgError)
print("错误序列来了")
count += 1
}
observer.onNext("S")
observer.onNext("E")
observer.onNext("N")
observer.onCompleted()
return Disposables.create()
}
sequenceThatErrors
.retry(3)
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
debug 操作符
打印所有订阅、事件和处理。
一、debug
var count = 1
let sequenceThatErrors = Observable<String>.create { observer in
observer.onNext("J")
observer.onNext("E")
observer.onNext("N")
if count < 5 {
observer.onError(self.lgError)
print("错误序列来了")
count += 1
}
observer.onNext("S")
observer.onNext("E")
observer.onNext("N")
observer.onCompleted()
return Disposables.create()
}
sequenceThatErrors
.retry(3)
.debug()
.subscribe(onNext: { print($0) })
.disposed(by: disposeBag)
链接操作符
publish
将源可观察序列转换为可连接序列,并通过指定的主题广播其发射。
let subject = PublishSubject<Any>()
subject.subscribe{print("00:\($0)")}
.disposed(by: disposeBag)
let netOB = Observable<Any>.create { (observer) -> Disposable in
sleep(2)// 模拟网络延迟
print("我开始请求网络了")
observer.onNext("请求到的网络数据")
observer.onNext("请求到的本地")
observer.onCompleted()
return Disposables.create {
print("销毁回调了")
}
}.publish()
netOB.subscribe(onNext: { (anything) in
print("订阅1:",anything)
})
.disposed(by: disposeBag)
// 我们有时候不止一次网络订阅,因为有时候我们的数据可能用在不同的额地方
// 所以在订阅一次 会出现什么问题?
netOB.subscribe(onNext: { (anything) in
print("订阅2:",anything)
})
.disposed(by: disposeBag)
_ = netOB.connect()
replay
将源可观察序列转换为可连接的序列,并将向每个新订阅服务器重放以前排放的缓冲大小
// 首先拥有和publish一样的能力,共享 Observable sequence, 其次使用replay还需要我们传入一个参数(buffer size)来缓存已发送的事件,当有新的订阅者订阅了,会把缓存的事件发送给新的订阅者
let interval = Observable<Int>.interval(.seconds(1), scheduler: MainScheduler.instance).replay(5)
interval.subscribe(onNext: { print(Date.time,"订阅: 1, 事件: \($0)") })
.disposed(by: self.disposeBag)
delay(2) { _ = interval.connect() }
delay(4) {
interval.subscribe(onNext: { print(Date.time,"订阅: 2, 事件: \($0)") })
.disposed(by: self.disposeBag)
}
delay(8) {
interval.subscribe(onNext: { print(Date.time,"订阅: 3, 事件: \($0)") })
.disposed(by: self.disposeBag)
}
delay(20, closure: {
self.disposeBag = DisposeBag()
})
`
网友评论