176 votes

Permutation des clés et des valeurs dans un hachage

En Ruby, comment échanger les clés et les valeurs d'un hachage ?

Disons que j'ai le hachage suivant :

{:a=>:one, :b=>:two, :c=>:three}

En qui je veux me transformer :

{:one=>:a, :two=>:b, :three=>:c}

L'utilisation d'une carte semble plutôt fastidieuse. Existe-t-il une solution plus courte ?

317voto

Nigel Thorne Points 6412

Hash#inversé devrait fonctionner dans la plupart des situations.

{a: 1, b: 2, c: 3}.invert
=> {1=>:a, 2=>:b, 3=>:c}

MAIS...

Si vous avez des valeurs en double, Invert éliminera toutes vos valeurs sauf la dernière.

{a: 1, b: 2, c: 2}.invert
=> {1=>:a, 2=>:c}

Donc si vos valeurs sont uniques vous pouvez utiliser Hash#invert Sinon, vous pouvez conserver toutes les valeurs dans un tableau, comme ceci :

class Hash
  # like invert but not lossy
  # {"one"=>1,"two"=>2, "1"=>1, "2"=>2}.inverse => {1=>["one", "1"], 2=>["two", "2"]} 
  def safe_invert
    each_with_object({}) do |(key,value),out| 
      out[value] ||= []
      out[value] << key
    end
  end
end

Note : Ce code avec les tests est maintenant ici .

Ou en bref...

class Hash
  def safe_invert
    self.each_with_object( {} ) { |(key, value), out| ( out[value] ||= [] ) << key }
  end
end

4 votes

each_with_object a plus de sens ici que inject .

0 votes

Ce qui donne each_with_object({}){ |i,o|k,v = *i; o[v] ||=[]; o[v] << k} ... sympa

3 votes

Omg. Je ne savais pas qu'on pouvait faire |(key,value),out|. C'est génial, je détestais que le tableau vienne à la place de key et value. Merci beaucoup

67voto

Jonathan Allard Points 3754

Vous pouvez être sûr qu'il y en a un ! Il y a toujours une façon plus courte de faire les choses en Ruby !

C'est assez simple, il suffit d'utiliser Hash#invert :

{a: :one, b: :two, c: :three}.invert
=> {:one=>:a, :two=>:b, :three=>:c}

Et voilà !

5 votes

Hash#invert ne fonctionne pas si les mêmes valeurs apparaissent plusieurs fois dans votre hachage.

3voto

Riaze Points 64
files = {
  'Input.txt' => 'Randy',
  'Code.py' => 'Stan',
  'Output.txt' => 'Randy'
}

h = Hash.new{|h,k| h[k] = []} # Create hash that defaults unknown keys to empty an empty list
files.map {|k,v| h[v]<< k} #append each key to the list at a known value
puts h

Cela permettra de gérer également les valeurs en double.

2voto

dawg Points 26051

Si vous avez un hachage où les clés sont uniques, vous pouvez utiliser Hash#inversé :

> {a: 1, b: 2, c: 3}.invert
=> {1=>:a, 2=>:b, 3=>:c} 

Cela ne fonctionnera pas si vous avez des clés non uniques, cependant, où seules les dernières clés vues seront conservées :

> {a: 1, b: 2, c: 3, d: 3, e: 2, f: 1}.invert
=> {1=>:f, 2=>:e, 3=>:d}

Si vous avez un hachage avec des clés non uniques, vous pouvez le faire :

> 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]}

Si les valeurs du hachage sont déjà des tableaux, vous pouvez le faire :

> 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"]}

1voto

Tilo Points 13833
# this doesn't looks quite as elegant as the other solutions here,
# but if you call inverse twice, it will preserve the elements of the original hash

# true inversion of Ruby Hash / preserves all elements in original hash
# e.g. hash.inverse.inverse ~ h

class Hash

  def inverse
    i = Hash.new
    self.each_pair{ |k,v|
      if (v.class == Array)
        v.each{ |x|
          i[x] = i.has_key?(x) ? [k,i[x]].flatten : k
        }
      else
        i[v] = i.has_key?(v) ? [k,i[v]].flatten : k
      end
    }
    return i
  end

end

Hash#inverse vous donne :

 h = {a: 1, b: 2, c: 2}
 h.inverse
  => {1=>:a, 2=>[:c, :b]}
 h.inverse.inverse
  => {:a=>1, :c=>2, :b=>2}  # order might not be preserved
 h.inverse.inverse == h
  => true                   # true-ish because order might change

alors que la fonction intégrée invert est tout simplement cassée :

 h.invert
  => {1=>:a, 2=>:c}    # FAIL
 h.invert.invert == h 
  => false             # FAIL

Prograide.com

Prograide est une communauté de développeurs qui cherche à élargir la connaissance de la programmation au-delà de l'anglais.
Pour cela nous avons les plus grands doutes résolus en français et vous pouvez aussi poser vos propres questions ou résoudre celles des autres.

Powered by:

X