可以观察到最近的价值

时间:2017-11-17 09:58:41

标签: kotlin rx-java system.reactive publishsubject

我实现了一个名为" FilterByLatestFrom"的伪运算符。作为kotlin的扩展函数。

我使用此运算符编写了下一个代码:

    fun testFilterByLatestFromOperator(){
    val observableA : Observable<Int> = Observable.fromArray(1,2,3,4,5,6,7,8,9,10)
    val observableC : PublishSubject<Int> = PublishSubject.create()
    val observableB : Observable<Int> = Observable.just(2).mergeWith(observableC)

    observableB.subscribe { println("observableB onNext: $it") }

    observableA
            .subscribe({ println("Original : $it")})

    observableA.filterByLatestFrom(observableB, BiFunction { aVal, bVal -> aVal%bVal==0 })
            .subscribe({ println("Result A : $it") })

    observableC.onNext(3)

    observableA.filterByLatestFrom(observableB, BiFunction { aVal, bVal -> aVal%bVal==0 })
            .subscribe({ println("Result AC : $it") })
}

输出是:

observableB onNext: 2
Original : 1
Original : 2
Original : 3
Original : 4
Original : 5
Original : 6
Original : 7
Original : 8
Original : 9
Original : 10
Result A : 2
Result A : 4
Result A : 6
Result A : 8
Result A : 10
observableB onNext: 3
Result AC : 2
Result AC : 4
Result AC : 6
Result AC : 8
Result AC : 10

我希望过滤器运算符根据可观察B的最后值过滤obsA。 它适用于第一个块,但是当我使用新值添加On-next时,它不会更改结果(使用原始observable中的相同最后一个值)。

这是FilterByLatestFrom impl(它的设计也是从Java中使用的(使用compose):

class FilterByLatestFrom<T,U>(private val observable: Observable<T>, private val biFunction: BiFunction<U, T, Boolean>){
fun filter() : ObservableTransformer<U,U> = ObservableTransformer {
    it
            .withLatestFrom(
                    observable,
                    BiFunction<U,T,Pair<U,Boolean>> {
                        u, t -> Pair(u,biFunction.apply(u,t))
                    })
            .filter { it.second }
            .map { it.first }
    }
}
fun <T,U> Observable<U>.filterByLatestFrom(observable: Observable<T>, biFunction: BiFunction<U, T, Boolean>) : Observable<U> =
        this.compose(FilterByLatestFrom(observable,biFunction).filter())

我错过了什么?

编辑:我认为我发现了问题:PublishSubject应该是BehaviorSubject。并且合并函数应该是在blindB之后承诺obsC将发出的。

1 个答案:

答案 0 :(得分:2)

你的伪运算符GetFrameworkPath很好,问题在于测试,filterByLatestFrom只会发出后续项目,所以当你在上次订阅时('结果AC'),{{ 1}}将仅发出2,因为PublishSubject已经发出3,并且不会将其重播为observableB(使用observableC)。

只需将observableB移到最后一个订阅(最后一行)之后,您就会看到预期的行为。

编辑: 也改为merge,就像你解决了同样的问题一样(主题会重播新订阅的最后一个值)