Превращение Хэша Массивов в Массив Хэшей в Рубине - PullRequest
11 голосов
/ 29 октября 2009

У нас есть следующие структуры данных:

{:a => ["val1", "val2"], :b => ["valb1", "valb2"], ...}

И я хочу превратить это в

[{:a => "val1", :b => "valb1"}, {:a => "val2", :b => "valb2"}, ...]

А потом вернемся к первому виду. Кто-нибудь с красивой реализацией?

Ответы [ 7 ]

12 голосов
/ 29 октября 2009

Это решение работает с произвольным числом значений (val1, val2 ... valN):

{:a => ["val1", "val2"], :b => ["valb1", "valb2"]}.inject([]){|a, (k,vs)| 
  vs.each_with_index{|v,i| (a[i] ||= {})[k] = v} 
  a
}
# => [{:a=>"val1", :b=>"valb1"}, {:a=>"val2", :b=>"valb2"}]

[{:a=>"val1", :b=>"valb1"}, {:a=>"val2", :b=>"valb2"}].inject({}){|a, h| 
  h.each_pair{|k,v| (a[k] ||= []) << v}
  a
}
# => {:a=>["val1", "val2"], :b=>["valb1", "valb2"]}
7 голосов
/ 23 января 2011

Используя функциональный подход (см. Enumerable):

hs = h.values.transpose.map { |vs| h.keys.zip(vs).to_h }
#=> [{:a=>"val1", :b=>"valb1"}, {:a=>"val2", :b=>"valb2"}]

и обратно:

h_again = hs.first.keys.zip(hs.map(&:values).transpose).to_h
#=> {:a=>["val1", "val2"], :b=>["valb1", "valb2"]}
1 голос
/ 07 августа 2014

Моя попытка, возможно, немного более компактная.

h = { :a => ["val1", "val2"], :b => ["valb1", "valb2"] }

h.values.transpose.map { |s| Hash[h.keys.zip(s)] }

Должно работать в Ruby 1.9.3 или новее.


Пояснение:

Сначала «объединить» соответствующие значения в «строки»

h.values.transpose
# => [["val1", "valb1"], ["val2", "valb2"]] 

Каждая итерация в блоке map будет производить один из них:

h.keys.zip(s)
# => [[:a, "val1"], [:b, "valb1"]]

и Hash[] превратят их в хэши:

Hash[h.keys.zip(s)]
# => {:a=>"val1", :b=>"valb1"}      (for each iteration)
1 голос
/ 29 октября 2009
m = {}
a,b = Array(h).transpose
b.transpose.map { |y| [a, y].transpose.inject(m) { |m,x| m.merge Hash[*x] }}
1 голос
/ 29 октября 2009

Давайте посмотрим, между какими структурами данных мы пытаемся преобразовать:

#Format A
[
 ["val1", "val2"],          :a
 ["valb1", "valb2"],        :b 
 ["valc1", "valc2"]         :c 
]
#Format B
[ :a        :b       :c
 ["val1", "valb1", "valc1"],
 ["val2", "valb2", "valc3"]
]

Нетрудно найти Format B - это транспонирование Format A в сущности, тогда мы можем прийти к такому решению:

h={:a => ["vala1", "vala2"], :b => ["valb1", "valb2"], :c => ["valc1", "valc2"]}
sorted_keys =  h.keys.sort_by {|a,b| a.to_s <=> b.to_s}

puts sorted_keys.inject([])  {|s,e| s << h[e]}.transpose.inject([])   {|r, a| r << Hash[*sorted_keys.zip(a).flatten]}.inspect
#[{:b=>"valb1", :c=>"valc1", :a=>"vala1"}, {:b=>"valb2", :c=>"valc2", :a=>"vala2"}]
0 голосов
/ 29 октября 2009

Это будет работать при условии, что все массивы в оригинальном хэше имеют одинаковый размер:

hash_array = hash.first[1].map { {} }
hash.each do |key,arr|
  hash_array.zip(arr).each {|inner_hash, val| inner_hash[key] = val}
end
0 голосов
/ 29 октября 2009

Вы можете использовать inject для построения массива хэшей.

hash = { :a => ["val1", "val2"], :b => ["valb1", "valb2"] }
array = hash.inject([]) do |pairs, pair|
  pairs << { pair[0] => pair[1] }
  pairs
end
array.inspect # => "[{:a=>["val1", "val2"]}, {:b=>["valb1", "valb2"]}]"

Документация Ruby содержит еще несколько примеров работы с inject.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...