要散列的哈希数组

问题描述 投票:58回答:6

例如,我有一些单个哈希值

a = [{a: :b}, {c: :d}]

将它转换成这个的最佳方法是什么?

{a: :b, c: :d}
ruby arrays hash
6个回答
109
投票

你可以用

a.reduce Hash.new, :merge

直接产生

{:a=>:b, :c=>:d}

请注意,在发生碰撞的情况下,订单很重要。后者哈希覆盖以前的映射,例如:

[{a: :b}, {c: :d}, {e: :f, a: :g}].reduce Hash.new, :merge   # {:a=>:g, :c=>:d, :e=>:f}

35
投票

你可以使用.inject

a.inject(:merge)
#=> {:a=>:b, :c=>:d}

Demonstration

它从两个合并的每次迭代中启动一个新的哈希。为了避免这种情况,你可以使用破坏性的:merge!(或:update,它是相同的):

a.inject(:merge!)
#=> {:a=>:b, :c=>:d}

Demonstration


21
投票

这两个:

total_hash = hs.reduce({}) { |acc_hash, hash| acc_hash.merge(hash) }
total_hash = hs.reduce({}, :merge)

请注意,Hash#merge会在每次迭代时创建一个新的哈希,如果您正在构建一个大哈希,这可能是一个问题。在这种情况下,请改用update

total_hash = hs.reduce({}, :update)

另一种方法是将哈希转换为对,然后构建最终哈希:

total_hash = hs.flat_map(&:to_a).to_h

1
投票

我遇到了这个答案,我想在性能方面比较两个选项,看看哪个更好:

  1. a.reduce Hash.new, :merge
  2. a.inject(:merge)

使用ruby基准模块,结果证明选项(2)a.inject(:merge)更快。

用于比较的代码:

require 'benchmark'

input = [{b: "c"}, {e: "f"}, {h: "i"}, {k: "l"}]
n = 50_000

Benchmark.bm do |benchmark|
  benchmark.report("reduce") do
    n.times do
      input.reduce Hash.new, :merge
    end
  end

  benchmark.report("inject") do
    n.times do
      input.inject(:merge)
    end
  end
end

结果是

       user     system      total        real
reduce  0.125098   0.003690   0.128788 (  0.129617)
inject  0.078262   0.001439   0.079701 (  0.080383)

0
投票

试试这个

a.inject({}){|acc, hash| acc.merge(hash)} #=> {:a=>:b, :c=>:d}

0
投票

只是用

a.reduce(:merge)
#=> {:a=>:b, :c=>:d}

0
投票

您可以将其转换为数组[[:a, :b]],之后将所有内容翻译为哈希{:a=>:b}

# it works like [[:a, :b]].to_h => {:a=>:b}

[{a: :b}, {c: :d}].map { |hash| hash.to_a.flatten }.to_h

# => {:a=>:b, :c=>:d}
© www.soinside.com 2019 - 2024. All rights reserved.