如何通过一组新的键将哈希表中所有键替换为新的键?
有没有一种优雅的方式可以完成这个操作?
假设你有一个将旧键映射到新键的Hash
,你可以这样做:
hsh.transform_keys(&key_map.method(:[]))
Ruby 2.5有Hash#transform_keys!方法。以下是使用键映射的示例:
h = {a: 1, b: 2, c: 3}
key_map = {a: 'A', b: 'B', c: 'C'}
h.transform_keys! {|k| key_map[k]}
# => {"A"=>1, "B"=>2, "C"=>3}
您也可以使用symbol#toproc快捷方式,配合transform_keys使用。例如:h.transform_keys! &:upcase
# => {"A"=>1, "B"=>2, "C"=>3}
keys
,而不更改值:hash = {
"nr"=>"123",
"name"=>"Herrmann Hofreiter",
"pferd"=>"010 000 777",
"land"=>"hight land"
}
header = ["aa", "bb", "cc", "dd"]
new_hash = header.zip(hash.values).to_h
结果:
{
"aa"=>"123",
"bb"=>"Herrmann Hofreiter",
"cc"=>"010 000 777",
"dd"=>"high land"
}
hash = {
'foo' => 1,
'bar' => 2
}
new_keys = {
'foo' => 'foozle',
'bar' => 'barzle'
}
new_keys.values.zip(hash.values_at(*new_keys.keys)).to_h
# => {"foozle"=>1, "barzle"=>2}
拆分一下:
new_keys
.values # => ["foozle", "barzle"]
.zip(
hash.values_at(*new_keys.keys) # => [1, 2]
) # => [["foozle", 1], ["barzle", 2]]
.to_h
# => {"foozle"=>1, "barzle"=>2}
到了测试时间了...
虽然我喜欢Jörn的简洁回答,但我不确定它是否达到了应该有的速度,然后我看到了selvamani的评论:
require 'fruity'
HASH = {
'foo' => 1,
'bar' => 2
}
NEW_KEYS = {
'foo' => 'foozle',
'bar' => 'barzle'
}
compare do
mittag { HASH.dup.map {|k, v| [NEW_KEYS[k], v] }.to_h }
ttm { h = HASH.dup; NEW_KEYS.values.zip(h.values_at(*NEW_KEYS.keys)).to_h }
selvamani { h = HASH.dup; h.keys.each { |key| h[NEW_KEYS[key]] = h.delete(key)}; h }
end
# >> Running each test 2048 times. Test will take about 1 second.
# >> selvamani is faster than ttm by 39.99999999999999% ± 10.0%
# >> ttm is faster than mittag by 10.000000000000009% ± 10.0%
这些选项的速度非常接近,因此任何一个都可以使用,但是39%的回报率随着时间的推移会更好,因此请考虑选择该选项。由于存在可能导致结果不佳的潜在缺陷,因此未包括其中的一些答案。
具体解决方案取决于您拥有新密钥的格式(或者是否可以从旧密钥派生新密钥)。
假设您有一个哈希表h
,其键需要修改,以及一个将当前键映射到新键的哈希表new_keys
,您可以执行以下操作:
h.keys.each do |key|
h[new_keys[key]] = h[key] # add entry for new key
k.delete(key) # remove old key
end
new_keys
意外地返回key
本身时,这将删除key
的值。barbolos在此问题的答案中克服了这个问题。 - sawah.keys.each { |key| h[new_keys[key]] = h.delete(key)}
替代此代码。 - Selvamanih = { 'foo'=>1, 'bar'=>2 }
key_map = { 'foo'=>'foozle', 'bar'=>'barzle' }
h.each_with_object({}) { |(k,v),g| g[key_map[k]]=v }
#=> {"foozle"=>1, "barzle"=>2}
或者
h.reduce({}) { |g,(k,v)| g.merge(key_map[k]=>v) }
#=> {"foozle"=>1, "barzle"=>2}
each_with_object
); caryewo与mittag相似; mittag比caryred(使用reduce
)快70.0% ± 10.0%”。 - Cary Swoveland如果你也担心性能问题,这个更快:
hsh.keys.each { |k| hsh[ key_map[k] ] = hsh.delete(k) if key_map[k] }
不需要创建新的哈希表,只需重命名必要的键即可提高性能。
您可以在 "如何优雅地在 Ruby 中重命名哈希表中的所有键?" 中找到更多详细信息。
key_map[k]
为空,则会出现错误。 - the Tin Man
key_map
,它是一个键/值对哈希表,其中键是旧键,值是要交换的新键。 - the Tin Man&key_map.method(:[])
是如何工作的吗? - buncis