J'ai un tableau de hachages comme ceci:
[{"testPARAM1"=>"testVAL1"}, {"testPARAM2"=>"testVAL2"}]
Et j'essaie de mapper cela sur un hachage unique comme ceci:
{"testPARAM2"=>"testVAL2", "testPARAM1"=>"testVAL1"}
Je l'ai atteint en utilisant
par={}
mitem["params"].each { |h| h.each {|k,v| par[k]=v} }
Mais je me demandais s'il était possible de le faire de manière plus idiomatique (de préférence sans utiliser de variable locale).
Comment puis-je faire ceci?
Vous pouvez composer Enumerable#reduce
Et Hash#merge
Pour accomplir ce que vous voulez.
input = [{"testPARAM1"=>"testVAL1"}, {"testPARAM2"=>"testVAL2"}]
input.reduce({}, :merge)
is {"testPARAM2"=>"testVAL2", "testPARAM1"=>"testVAL1"}
Réduire une sorte de tableau comme coller un appel de méthode entre chaque élément de celui-ci.
Par exemple, [1, 2, 3].reduce(0, :+)
revient à dire 0 + 1 + 2 + 3
Et donne 6
.
Dans notre cas, nous faisons quelque chose de similaire, mais avec la fonction de fusion, qui fusionne deux hachages.
[{:a => 1}, {:b => 2}, {:c => 3}].reduce({}, :merge)
is {}.merge({:a => 1}.merge({:b => 2}.merge({:c => 3})))
is {:a => 1, :b => 2, :c => 3}
Que diriez-vous:
h = [{"testPARAM1"=>"testVAL1"}, {"testPARAM2"=>"testVAL2"}]
r = h.inject(:merge)
Utilisez # inject
hashes = [{"testPARAM1"=>"testVAL1"}, {"testPARAM2"=>"testVAL2"}]
merged = hashes.inject({}) { |aggregate, hash| aggregate.merge hash }
merged # => {"testPARAM1"=>"testVAL1", "testPARAM2"=>"testVAL2"}
Ici, vous pouvez utiliser la classe inject ou réduire de Enumerable car les deux sont des alias l'un de l'autre, il n'y a donc aucun avantage en termes de performances.
sample = [{"testPARAM1"=>"testVAL1"}, {"testPARAM2"=>"testVAL2"}]
result1 = sample.reduce(:merge)
# {"testPARAM1"=>"testVAL1", "testPARAM2"=>"testVAL2"}
result2 = sample.inject(:merge)
# {"testPARAM1"=>"testVAL1", "testPARAM2"=>"testVAL2"}