【发布时间】:2012-04-12 18:47:15
【问题描述】:
我在 Groovy [a: 1, b: 2] 和 [b:1, c:3] 中有两张地图,我想从它们中创建第三张地图 [a: 1, b: 3, c: 3]。是否有执行此操作的 Groovy 命令?
编辑:请注意,如果键相同,则第三个映射中的值是前两个映射中的值的总和。
谢谢
【问题讨论】:
标签: dictionary groovy
我在 Groovy [a: 1, b: 2] 和 [b:1, c:3] 中有两张地图,我想从它们中创建第三张地图 [a: 1, b: 3, c: 3]。是否有执行此操作的 Groovy 命令?
编辑:请注意,如果键相同,则第三个映射中的值是前两个映射中的值的总和。
谢谢
【问题讨论】:
标签: dictionary groovy
另一种解决方案是:
def m1 = [ a:1, b:2 ]
def m2 = [ b:1, c:3 ]
def newMap = [m1,m2]*.keySet().flatten().unique().collectEntries {
[ (it): [m1,m2]*.get( it ).findAll().sum() ]
}
以epidemian's answer为灵感,你也可以写一个处理多张地图的方法
def m1 = [a: 1, b: 2]
def m2 = [b: 1, c: 3]
def combine( Map... m ) {
m.collectMany { it.entrySet() }.inject( [:] ) { result, e ->
result << [ (e.key):e.value + ( result[ e.key ] ?: 0 ) ]
}
}
def newMap = combine( m1, m2 )
【讨论】:
这应该适用于任意数量的地图:
def maps = [[a: 1, b: 2], [b:1, c:3]]
def result = [:].withDefault{0}
maps.collectMany{ it.entrySet() }.each{ result[it.key] += it.value }
assert result == [a: 1, b: 3, c: 3]
maps.collectMany{ it.entrySet() } 表达式返回映射条目列表,例如 [a=1, b=2, b=1, c=3],然后将每个条目添加到结果中。
如果您想将所有转换保留在一个表达式中并使其“更具功能性”,另一种选择是首先按键对条目进行分组,然后对值求和,但我认为它的可读性较差:
def result = maps.collectMany{ it.entrySet() }
.groupBy{ it.key }
.collectEntries{[it.key, it.value.sum{ it.value }]}
groupBy 部分返回 [a:[a=1], b:[b=2, b=1], c:[c=3]] 形式的映射,然后 collectEntries 将该映射转换为另一个具有相同 kays 但值中包含列表总和的映射。
【讨论】:
collectMany(启发我写了一个可变参数方法)
foo.bar { ... }.baz { ... } 这样的链式调用时,方法名称和闭包的左大括号之间应该有一个空格,而不是在右大括号和下一个点运算符?我觉得写答案的时候看起来很奇怪,所以我删除了第一个空格,但它看起来也不太好......
以下示例演示了将两个映射添加到第三个映射m3:
Map m1 = [ a:1, b:2 ];
Map m2 = [ b:1, c:3 ];
Map m3 = [:];
m3 << m1
m3 << m2
【讨论】:
一个很好的方法是使用扩展运算符:
def m1 = [ a:1, b:2 ]
def m2 = [ b:1, c:3 ]
def m3 = [ *:m1, *:m2 ]
println "m1 = $m1"
println "m2 = $m2"
println "m3 = $m3"
*信用:http://mrhaki.blogspot.ch/2009/09/groovy-goodness-spread-operator.html
上面的代码在 groovysh 中不起作用,但在 groovy 控制台中很好。 (适用于 1.8 版)
【讨论】:
我认为没有现成的方法,也许使用类似的方法:
def m1 = [a: 1, b: 2]
def m2 = [b: 1, c: 3]
def newMap = (m1.keySet() + m2.keySet()).inject([:]) {
acc, k -> acc[k] = (m1[k] ?: 0) + (m2[k] ?: 0); acc
}
【讨论】:
这样做:
Map additionJoin( Map map1, Map map2 )
{
def result = [:];
result.putAll( map1 );
result.putAll( map2 );
result.each { key, value ->
if( map1[key] && map2[key] )
{
result[key] = map1[key] + map2[key]
}
}
return result;
}
def a = [a: 1, b: 2]
def b = [b:1,c:3]
def c = additionJoin( a, b )
println c
【讨论】:
def merge(map1, map2) {
def add = { map, entry -> map << entry }
map2.inject(map1.inject([:], add), add)
}
【讨论】:
a 并且不从两个映射中为键 b 添加值