2011-10-24 15 views
8

Mam skrót, w którym chcę użyć wartości jako kluczy w nowym haszowniku, który zawiera liczbę, ile razy ten element pojawił się jako wartość w oryginalnym haszu.Metoda "zliczania" ruby ​​dla skrótów

więc używam:

hashA.keys.each do |i| 
    puts hashA[i] 
end 

Przykâadowa:

0 
1 
1 
2 
0 
1 
1 

I chcę nowy Hash być następujące:

{ 0 => 2, 1 => 4, 2 => 1 } 
+0

możliwy duplikat [Jak liczyć identyczne elementy ciągu w tablicy Ruby] (http://stackoverflow.com/questions/5128200/how-to-count-identical-string-elements-in-a-ruby -szyk). Oryginalna struktura danych w tym pytaniu jest hash, ale wyrzucasz klucze, więc skutecznie radzisz sobie z 'hashA.values', które jest tablicą. –

+0

@AndrewGrimm Meh; ostatecznie (i szybko), tak ... Ale podczas wyszukiwania z nastawieniem "Mam mapę" prawdopodobnie nie będziesz szukać rzeczy o tablicach. –

Odpowiedz

17
counts = hashA.values.inject(Hash.new(0)) do |collection, value| 
    collection[value] +=1 
    collection 
end 
+5

+1 dla znaczących nazw zmiennych w bloku; Zapominam, kiedy nie znasz API, mają znaczenie. –

7

TL; DR: hashA.values.inject(Hash.new(0)) { |m, n| m[n] += 1; m }

> hashA = { a: 0, b: 1, c: 1, d: 2, e: 0, f: 1, g: 1 } 
=> {:a=>0, :b=>1, :c=>1, :d=>2, :e=>0, :f=>1, :g=>1} 
> hashCounts = hashA.values.inject(Hash.new(0)) { |m, n| m[n] += 1; m } 
=> {0=>2, 1=>4, 2=>1}