这是对哈希进行排序并返回hash对象(而不是Array)的最佳方法吗?

h = {"a"=>1, "c"=>3, "b"=>2, "d"=>4}
# => {"a"=>1, "c"=>3, "b"=>2, "d"=>4}

Hash[h.sort]
# => {"a"=>1, "b"=>2, "c"=>3, "d"=>4}

当前回答

不,它不是(Ruby 1.9.x)

require 'benchmark'

h = {"a"=>1, "c"=>3, "b"=>2, "d"=>4}
many = 100_000

Benchmark.bm do |b|
  GC.start

  b.report("hash sort") do
    many.times do
      Hash[h.sort]
    end
  end

  GC.start

  b.report("keys sort") do
    many.times do
      nh = {}
      h.keys.sort.each do |k|
        nh[k] = h[k]
      end
    end
  end
end

       user     system      total        real
hash sort  0.400000   0.000000   0.400000 (  0.405588)
keys sort  0.250000   0.010000   0.260000 (  0.260303)

对于较大的散列,差异将增长到10倍甚至更多

其他回答

我一直使用sort_by。您需要用Hash[]包装#sort_by输出以使其输出散列,否则它将输出数组的数组。另外,要实现这一点,您可以在元组数组上运行#to_h方法,将它们转换为k=>v结构(哈希)。

hsh ={"a" => 1000, "b" => 10, "c" => 200000}
Hash[hsh.sort_by{|k,v| v}] #or hsh.sort_by{|k,v| v}.to_h

在“如何根据数值对Ruby哈希进行排序?”中有一个类似的问题。

你在OP中给了自己最好的答案:Hash[h]。如果你渴望更多的可能性,这里是原始哈希的就地修改,使其排序:

h.keys.sort.each { |k| h[k] = h.delete k }

我借用Boris Stitnicky的灵感解决方案来修补一个就地排序!方法:

def sort!
  keys.sort!.each { |k| store k, delete(k) }
  self
end

在Ruby 2.1中,它很简单:

h.sort.to_h

不,它不是(Ruby 1.9.x)

require 'benchmark'

h = {"a"=>1, "c"=>3, "b"=>2, "d"=>4}
many = 100_000

Benchmark.bm do |b|
  GC.start

  b.report("hash sort") do
    many.times do
      Hash[h.sort]
    end
  end

  GC.start

  b.report("keys sort") do
    many.times do
      nh = {}
      h.keys.sort.each do |k|
        nh[k] = h[k]
      end
    end
  end
end

       user     system      total        real
hash sort  0.400000   0.000000   0.400000 (  0.405588)
keys sort  0.250000   0.010000   0.260000 (  0.260303)

对于较大的散列,差异将增长到10倍甚至更多