背景
我在单例管理类里按如下代码的方式声明了observable1,本希望每个地方都可以订阅该唯一信号,结果发现除第一次订阅以外,后面的订阅经常不执行。
var observable1: Observable<(String?, String?)> {
return Observable.combineLatest(subject1, subject2)
}
下图为combineLatest运算符的特性
![](https://img-blog.csdnimg.cn/img_convert/701044431b9e7bcf721c28e140d40476.png)
图1
实战举例
下图中打印了三次observable1,每次地址均不相同。原因是每次get observable1都会return一个新对象导致的。
![](https://img-blog.csdnimg.cn/img_convert/d15e8ccb69fa8afb122d34be3b01f2f2.png)
图2
由于combineLatest的特性是必须每个subject都产生过信号,才会在任何一个subject发出信号时,把两个subject的最新的元素组合后作为新的信号发出去。因此后面订阅的observable1,由于是新的对象,所以可能不满足两个subject都产生过信号,也就是订阅新的observable1之后,只有某一个subject发出信号,并不会触发订阅,因为另一个subject不存在最新的元素。如下图打印所示:
![](https://img-blog.csdnimg.cn/img_convert/546c3b91f33937515531a04dbb176f62.png)
图3
解决办法
方案一
如下图,初始化时创建observable1,后面不论订阅几次,都只有一个observable1。根本原因还是避免了每次get时return一个新的对象。
![](https://img-blog.csdnimg.cn/img_convert/b552194a85e32165b926d44f6c7254ed.png)
图4
方案二
observable1改为PublishSubject类型,记录subject1和subject2的最新元素,在两个subject任何一个发出信号时,取出两个subject的最新元素同时发出observable1信号。或者两个subject采用BehaviorRelay类型 ,直接读value就是最新元素,在两个BehaviorRelay信号中任何一个accept新元素后,observable1发出组合后的元素。