2011-01-14 62 views
11

我不知道是否有在红宝石做到这一点更规范的方式的项目的属性的对象数组1.9大厦哈希通过分组基于

我有一大堆对象的数组,我想使用数组中每个对象的属性将它们组成一个Hash。

非常简单的例子:

> sh = {} 
=> {} 
> aers = %w(a b c d ab bc de abc) 
=> ["a", "b", "c", "d", "ab", "bc", "de", "abc"] 
> aers.each do |aer| 
>  sh[aer.size] = [] if sh[aer.size].nil? 
>  sh[aer.size] << aer 
> end 
=> ["a", "b", "c", "d", "ab", "bc", "de", "abc"] 
> sh 
=> {1=>["a", "b", "c", "d"], 2=>["ab", "bc", "de"], 3=>["abc"]} 

我想这一点,但它的输出是错误的(因为你可以看到):

sh = Hash.new([]) 
=> {} 
> aers.each do |aer| 
>  sh[aer.size] << aer 
> end 
=> ["a", "b", "c", "d", "ab", "bc", "de", "abc"] 
> sh 
=> {} 
+2

而你的代码没有按预期工作的原因说明如下:http://stackoverflow.com/questions/2698460/strange-ruby-behavior-when-using -hash-new和here:http://stackoverflow.com/questions/2552579/ruby-method-array-not-updating-the-array-in-hash(Ruby中很常见的陷阱)。 – 2011-01-14 15:23:51

回答

30

红宝石已经预见你的需求,并且已经得到了你覆盖着Enumerable#group_by

irb(main):001:0> aers = %w(a b c d ab bc de abc) 
#=> ["a", "b", "c", "d", "ab", "bc", "de", "abc"] 

irb(main):002:0> aers.group_by{ |s| s.size } 
#=> {1=>["a", "b", "c", "d"], 2=>["ab", "bc", "de"], 3=>["abc"]} 

In R uby 1.9,你可以让这甚至更短:

irb(main):003:0> aers.group_by(&:size) 
#=> {1=>["a", "b", "c", "d"], 2=>["ab", "bc", "de"], 3=>["abc"]} 
+3

这就是为什么Ruby非常酷;它甚至在我们做之前就知道我们想要什么。 – 2011-01-14 23:42:08

1

您还可以通过链接的方法......这可能仅仅是为这个问题的学术兴趣的做到这一点,但仍然是一个很好的技术熟悉。

irb(main):017:0> sh = {} 
=> {} 
irb(main):018:0> aers.collect{|k| k.size}.uniq!.each{|k| sh[k] = aers.select{|j| j.size == k}} 
=> [1, 2, 3] 
irb(main):019:0> sh 
=> {1=>["a", "b", "c", "d"], 2=>["ab", "bc", "de"], 3=>["abc"]} 
irb(main):020:0> 
3

Phrogz is correct,group_by is there for the taking。你的代码包含了ruby的陷阱之一。

aers = %w(a b c d ab bc de abc) 
sh = Hash.new([]) # returns the _same_ array everytime the key is not found. 
# sh = Hash.new{|h,v| h[v] = []} # This one works 
p sh, sh.default 

aers.each do |aer| 
    sh[aer.size] << aer #modifies the default [] every time 
end 
p sh, sh.default 
p sh[5] 

输出

{} 
[] 
{} 
["a", "b", "c", "d", "ab", "bc", "de", "abc"] 
["a", "b", "c", "d", "ab", "bc", "de", "abc"]