你如何在Groovy中一起使用GroupBy和Sum?

Zac*_*ons 5 groovy

我有一个这样的集合:

[[patient1,value1], [patient2,value2]]
Run Code Online (Sandbox Code Playgroud)

例如.
x = [[1, 20.28], [1, 11.11], [2, 4.60], [2, 3.68]]

我用它countBy来得到每位患者的计数:

def counts = x.countBy{patient, value -> patient}
Run Code Online (Sandbox Code Playgroud)

我试图得到每个病人的总和没有运气:

def sums = x.groupBy({patient, value -> patient }).collectEntries{p, v -> p:v.sum()}
Run Code Online (Sandbox Code Playgroud)

有什么我想念的吗?

注意:这里的总体目标是获得患者使用的平均值:

def avgs = sums.collect{patient, value -> (value / counts[patient]) }
Run Code Online (Sandbox Code Playgroud)

谢谢!

Nat*_*hes 7

在groupBy方法中,闭包标识要分组的内容,列表条目在分组值下归档:

x.groupBy { it[0] } // map entries by patient number
Run Code Online (Sandbox Code Playgroud)

评估为

[1:[[1, 20.28], [1, 11.11]], 2:[[2, 4.60], [2, 3.68]]]
Run Code Online (Sandbox Code Playgroud)

collectEntries方法指定每个映射条目的外观,假设collectEntries的每个输入都有一个键的患者编号和该键的值对的列表.所以

x.groupBy {it[0]}.collectEntries {[(it.key): it.value.sum {it[1]}]}
Run Code Online (Sandbox Code Playgroud)

评估为

[1:31.39, 2:8.28]
Run Code Online (Sandbox Code Playgroud)

每位患者的平均值是

x.groupBy {it[0]}.collectEntries {[(it.key): it.value.sum {it[1]}/it.value.size()]}
Run Code Online (Sandbox Code Playgroud)

评估到 [1:15.695, 2:4.14]

平均值:

def avgs = [1:15.695, 2:4.14]
def averageOfAverages = avgs.entrySet().sum {it.value} / avgs.size()
Run Code Online (Sandbox Code Playgroud)