해시에서 키와 값을 바꾸는 방법


154

해시에서 키와 값을 어떻게 교환합니까?

다음과 같은 해시가 있습니다.

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

나는 다음과 같이 변형하고 싶다.

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

사용 map이 다소 지루한 것 같습니다. 더 짧은 해결책이 있습니까?

답변:


280

루비에는 해시를위한 도우미 메소드가 있습니다.이를 통해 해시가 거꾸로 된 것처럼 취급 할 수 있습니다 (본질적으로 값을 통해 키에 액세스 할 수 있도록하여).

{a: 1, b: 2, c: 3}.key(1)
=> :a

역 해시를 유지하려면 대부분의 상황에서 Hash # invert 가 작동합니다.

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

그러나...

중복 값이있는 경우 invert마지막으로 나타나는 값을 제외한 모든 값 을 버립니다 (반복 중에 해당 키의 새 값을 계속 대체하기 때문에). 마찬가지로 key첫 번째 일치 항목 만 반환합니다.

{a: 1, b: 2, c: 2}.key(2)
=> :b

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

따라서 값이 고유하면을 사용할 수 있습니다 Hash#invert. 그렇지 않은 경우 다음과 같이 모든 값을 배열로 유지할 수 있습니다.

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

참고 : 테스트가있는이 코드는 이제 GitHub에 있습니다 .

또는:

class Hash
  def safe_invert
    self.each_with_object({}){|(k,v),o|(o[v]||=[])<<k}
  end
end

4
each_with_object보다 더 의미가 있습니다 inject.
Andrew Marshall

그래서 each_with_object({}){ |i,o|k,v = *i; o[v] ||=[]; o[v] << k}... 좋은됩니다
Nigel Thorne

3
세상에 나는 당신이 | (key, value), out |을 할 수 있다는 것을 몰랐습니다. 너무 훌륭해서 키와 값 대신 배열이 들어오는 것을 싫어했습니다. 대단히 감사합니다
Iuri G.

63

당신은 하나가 내기! 루비에서 작업을 수행하는 짧은 방법이 항상 있습니다!

꽤 간단합니다 Hash#invert.

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

vo!


4
해시에 동일한 값이 여러 번 나타나는 경우 해시 #invert가 작동하지 않습니다.
Tilo

2
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

이것은 중복 값도 처리합니다.


1
각 단계에서 무슨 일이 일어나고 있는지에 대한 답을 조금 설명해 주시겠습니까?
Sajjad Murtaza

개인적으로 해시의 기본값을 설정하지 마십시오. 이 해시를 제공하는 코드가 해시가 그런 식으로 작동하지 않을 것이라고 걱정하고 나중에 교묘 한 오류가 발생할 수 있습니다. 나는이 문제를 정말로 정당화 할 수 없다. 무시할 수없는 것 같습니다. 최소한의 놀라움의 원리?
Nigel Thorne

코드로 응답 할 때 코드의 작동 방식과 적절한 솔루션 인 이유를 설명하는 것이 중요합니다. 목표는 단지 ​​즉각적인 문제를 해결하는 것이 아니라 교육하는 것입니다.
틴 남자

1
# 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 당신에게 제공합니다 :

 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

반면 내장 invert메소드는 깨졌습니다.

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

1

배열 사용

input = {:key1=>"value1", :key2=>"value2", :key3=>"value3", :key4=>"value4", :key5=>"value5"}
output = Hash[input.to_a.map{|m| m.reverse}]

해시 사용

input = {:key1=>"value1", :key2=>"value2", :key3=>"value3", :key4=>"value4", :key5=>"value5"}
output = input.invert

1

키가 고유 한 해시가 있는 경우 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"]}
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.