해시 배열을 단일 해시로 매핑하는 Rails


93

다음과 같은 해시 배열이 있습니다.

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

그리고 나는 이것을 다음과 같이 단일 해시에 매핑하려고합니다.

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

나는 그것을 사용하여 달성했습니다

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

하지만 좀 더 관용적 인 방식으로 (지역 변수를 사용하지 않고) 이것을 할 수 있는지 궁금합니다.

어떻게 할 수 있습니까?

답변:


164

원하는 것을 작곡 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}

1
감사합니다, 이것은 훌륭한 답변입니다 :) 아주 잘 설명되었습니다!
Bart Platak 2012 년

42
input.reduce (& : merge)면 충분합니다.
redgetan

@redgetan과 다른 점이 input.reduce(:merge)있습니까?
David van Geest

1
@David van Geest :이 경우에는 동등합니다. 여기에 사용 된 단항 앰퍼샌드는 기호에서 블록을 만듭니다. 그러나 reduce에는 기호를 허용하는 특별한 경우가 있습니다. 예제를 단순화하기 위해 단항 앰퍼샌드 연산자를 피하고 싶었지만이 경우 초기 값이 선택 사항이라는 redgetan이 맞습니다.
cjhveal

1
merge!대신 사용 merge하면 첫 번째 해시 (원하지 않을 수 있음)가 수정되지만 새로운 병합마다 중간 해시가 생성되지 않습니다.
Phrogz

50

어때 :

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

이 체계는 Joshua가 대답 한 것과 실질적으로 동일하지만 모든 해시에 #merge (기호로 전달 된 메서드 이름)를 반복적으로 적용합니다 (항목 사이에 연산자를 삽입하는 것으로 삽입하는 것을 생각해보십시오). #inject를 참조하십시오 .
shigeya

2
h.inject (& : merge)처럼 앰퍼샌드가 필요하지 않은 이유는 무엇입니까?
Donato

5
인 젝트 메소드는 메소드 이름으로 해석되는 매개 변수로 기호를 허용하기 때문입니다. 주입의 특징입니다.
shigeya

9

#inject 사용

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

0

여기 에서 Enumerable 클래스 에서 주입 또는 축소 를 사용할 수 있습니다. 둘 다 서로의 별칭이므로 둘 중 어느 쪽에도 성능상의 이점이 없습니다.

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

 result1 = sample.reduce(:merge)
 # {"testPARAM1"=>"testVAL1", "testPARAM2"=>"testVAL2"}

 result2 = sample.inject(:merge)
 # {"testPARAM1"=>"testVAL1", "testPARAM2"=>"testVAL2"}
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.