8

我得到了Observable<Rates>,Rate 只是一个简单的对象:

Rate(val value:String){}
Rates(val rates: List<Rate>)

我想把它Observable<Rates>改成Observable<HashMap<String,Long>.

因此,例如对于Rates(arrayOf(Rate("1"),Rate("2"), Rate("3"),Rate("3"), Rate("2"),Rate("2")))我期望结果的费率:

(1 -> 1)
(2 -> 3)
(3 -> 2)
(4 -> 0)
(5 -> 0)

我开始创建类似的东西:

service.getRates()
        .flatMap {it-> Observable.from(it.rates) }
        .filter { !it.value.isNullOrEmpty() }
        .groupBy {it -> it.value}
        .collect({ HashMap<String,Long>()}, { b, t -> b.put(t.key, t.count???)}

但我被困在这里,我不知道计算所有值?如果没有 5 of 4,我不知道如何添加空值 (0)。有没有办法使用 rx 来做到这一点?

4

3 回答 3

5

请查看代码中的注释以获取问题的答案。

import rx.Observable

fun main(args: Array<String>) {
    val service = Service()

    // This adds all keys with each key mapped to zero
    val referenceKeyCounts = Observable
        .just("1", "2", "3", "4", "5")
        .map { it to 0 }

    val keyCountsFromService = service.getRates()
        .flatMap { Observable.from(it.rates) }
        .filter { !it.value.isNullOrEmpty() }
        .map { it.value to 1 } // map each occurrence of key to 1

    Observable.concat(referenceKeyCounts, keyCountsFromService)
        .groupBy { it.first }
        .flatMap { group ->  // this converts GroupedObservable to final values
            group.reduce(0, { acc, pair -> acc + pair.second }) // add instead of counting
                .map { group.key to it }
        }
        .subscribe(::println)

}

class Service {
    fun getRates(): Observable<Rates> = Observable.just(Rates(listOf(
        Rate("1"), Rate("2"), Rate("3"), Rate("3"), Rate("2"), Rate("2")
    )))
}

class Rate(val value: String)

class Rates(val rates: List<Rate>)
于 2016-10-31T08:02:45.163 回答
4

诀窍是使用counton ,GroupedObservable因为它只在源 observable 完成时发出单个值:

在此处输入图像描述

从那里开始:

rates
  .flatMap { Observable.from(it.rates) }
  .filter { !it.value.isNullOrEmpty() }
  .groupBy { it.value }
  .flatMap { group -> group.count().map { group.key to it } } // list "1"->1, "2"->3, ...
  .mergeWith(Observable.from((1..5).map { it.toString() to 0 })) // defaults "4"->0
  .reduce(mutableMapOf<String, Int>()) { acc, cur ->
      acc.apply {
        val (key, count) = cur  
        this[key] = (this[key] ?: 0) + count // add counts
      }
  }.subscribe { countedRates -> 
    println(countedRates)
}
于 2016-10-31T06:14:45.970 回答
1

我认为这更多是关于函数式编程而不是与 RxJava 相关的问题

从 实现映射函数Rates -> Map<String,Int>

技巧:合并两个列表Pair<String,Int>以形成一个Map<String,Int>

val ratesToMapWithEmptyValues: (Rates) -> Map<String, Int> = { source ->

  //TODO: Just for demo
  val validRatesValue = arrayOf("1","2","3","4","5")

  mapOf(
    *validRatesValue.map { it to 0 }.toTypedArray(),
    *source.rates.groupBy(Rate::value).mapValues { it.value.size }.toList().toTypedArray()
  )

}

将函数应用在Observable.map

service.getRates()
        .map(ratesToMapWithEmptyValues)
于 2016-10-31T08:57:04.820 回答