Рельсы отображение массива хэшей на один хэш


у меня есть массив хэшей, например:

 [{"testPARAM1"=>"testVAL1"}, {"testPARAM2"=>"testVAL2"}]

и я пытаюсь сопоставить это на один хэш вроде этого:

{"testPARAM2"=>"testVAL2", "testPARAM1"=>"testVAL1"}

я достиг этого с помощью

  par={}
  mitem["params"].each { |h| h.each {|k,v| par[k]=v} } 

но мне было интересно, можно ли сделать это более идиоматическим способом (предпочтительно без использования локальной переменной).

как я могу это сделать?

3 69

3 ответа:

вы могли бы составить Enumerable#reduce и Hash#merge для достижения того, что вы хотите.

input = [{"testPARAM1"=>"testVAL1"}, {"testPARAM2"=>"testVAL2"}]
input.reduce({}, :merge)
  is {"testPARAM2"=>"testVAL2", "testPARAM1"=>"testVAL1"}

уменьшение массива вроде как прилипание вызова метода между каждым его элементом.

[1, 2, 3].reduce(0, :+) это как сказать 0 + 1 + 2 + 3 и дает 6.

в нашем случае мы делаем что-то подобное, но с функцией слияния, которая объединяет два хэша.

[{: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}

Как насчет:

h = [{"testPARAM1"=>"testVAL1"}, {"testPARAM2"=>"testVAL2"}]
r = h.inject(:merge)

использовать #inject

hashes = [{"testPARAM1"=>"testVAL1"}, {"testPARAM2"=>"testVAL2"}]
merged = hashes.inject({}) { |aggregate, hash| aggregate.merge hash }
merged # => {"testPARAM1"=>"testVAL1", "testPARAM2"=>"testVAL2"}