Если у вас есть хэш, где находятся ключи уникальны, вы можете использовать Hash#invert:
> {a: 1, b: 2, c: 3}.invert
=> {1=>:a, 2=>:b, 3=>:c}
Это не будет работать, если у вас есть не уникальные ключи, однако, где только последние ключи видно будет продолжал:
> {a: 1, b: 2, c: 3, d: 3, e: 2, f: 1}.invert
=> {1=>:f, 2=>:e, 3=>:d}
Если у вас есть хэш с не уникальным ключом, вы можете сделать:
> hash={a: 1, b: 2, c: 3, d: 3, e: 2, f: 1}
> hash.each_with_object(Hash.new { |h,k| h[k]=[] }) {|(k,v), h|
h[v] << k
}
=> {1=>[:a, :f], 2=>[:b, :e], 3=>[:c, :d]}
Если значения хэша уже являются массивами, вы можете сделать:
> hash={ "A" => [14, 15, 16], "B" => [17, 15], "C" => [35, 15] }
> hash.each_with_object(Hash.new { |h,k| h[k]=[] }) {|(k,v), h|
v.map {|t| h[t] << k}
}
=> {14=>["A"], 15=>["A", "B", "C"], 16=>["A"], 17=>["B"], 35=>["C"]}
Очень актуально! Спасибо за указание, я не проверял эти случаи. –
['each_with_object'] (http://ruby-doc.org/core-1.9.3/Enumerable.html#method-i-each_with_object) здесь имеет больше смысла, чем' inject'. –
, так что это будет 'each_with_object ({}) {| i, o | k, v = * i; o [v] || = []; o [v] << k} '... nice –