6

これらのオブジェクトのいくつかのプロパティに従ってソートされたオブジェクトの配列があります。優先順位の高い順に、これらのプロパティは、、fooおよびbarですbaz。これは、オブジェクトが最初にfoo;でソートされることを意味します。foo次に、同じ値を持つサブシーケンスがbar;でソートされます。次に、同じ値を持つものfooは。barでソートされbazます。

これを、このグループ化を反映するネストされたハッシュに変換したいと思います。基本的に私は再帰を探していEnumerable#group_byます。キーは、、、および;の値にfooなりbarますbaz。値は、オブジェクトのサブハッシュまたは配列のいずれかになります。次に例を示します。

[obj1, obj2, ... objn].group_by_recursive(:foo, :bar, :baz)
#=> {
      foo_val_1 => {
        bar_val_1 => {
          baz_val_1 => [
            obj1,
            obj2,
            obj3
          ],
          baz_val_2 => [
            obj4,
            obj5
          ]
        },
        bar_val_2 => {
          baz_val_1 => [
            obj6,
            obj7
          ],
          baz_val_2 => [
            obj8
          ]
        },
      },
      foo_val_2 => {
        ...
      },
      ...
    }
4

2 に答える 2

12

かなり良い解決策を思いついた。このようなモンキーパッチEnumerable

module Enumerable

  def group_by_recursive(*props)
    groups = group_by(&props.first)
    if props.count == 1
      groups
    else
      groups.merge(groups) do |group, elements|
        elements.group_by_recursive(*props.drop(1))
      end
    end
  end

end

渡すプロパティは、ProcsまたはSymbols

于 2013-03-11T11:47:04.133 に答える
1

ショーンと同様で、エラー処理がありません...

class Array
  def nested_group_by(*keys)
    return self if keys.length == 0
    groups = group_by(&keys.shift)
    Hash[groups.map { | k, v | [k, v.nested_group_by(*keys)] }]
  end
end
于 2013-03-11T11:58:07.060 に答える