我有一个条件,我得到一个哈希值
hash = {"_id"=>"4de7140772f8be03da000018", .....}
我想让这个散列
hash = {"id"=>"4de7140772f8be03da000018", ......}
附注:我不知道哈希中的键是什么,它们是随机的,每个键都有一个“_”前缀,我不想要下划线
我有一个条件,我得到一个哈希值
hash = {"_id"=>"4de7140772f8be03da000018", .....}
我想让这个散列
hash = {"id"=>"4de7140772f8be03da000018", ......}
附注:我不知道哈希中的键是什么,它们是随机的,每个键都有一个“_”前缀,我不想要下划线
当前回答
准确地回答了问题:
hash = {"_id"=>"4de7140772f8be03da000018"}
hash.transform_keys { |key| key[1..] }
# => {"id"=>"4de7140772f8be03da000018"}
从Ruby 2.5版开始,transform_keys方法就存在于Hash类中。
https://blog.bigbinary.com/2018/01/09/ruby-2-5-adds-hash-transform_keys-method.html
其他回答
如果你在一个哈希中有一个哈希,比如
hash = {
"object" => {
"_id"=>"4de7140772f8be03da000018"
}
}
如果你想把_id改成token
你可以在这里使用deep_transform_keys并这样做
hash.deep_transform_keys do |key|
key = "token" if key == "_id"
key
end
结果是
{
"object" => {
"token"=>"4de7140772f8be03da000018"
}
}
即使你有一个符号键哈希作为开始,比如
hash = {
object: {
id: "4de7140772f8be03da000018"
}
}
您可以组合所有这些概念,将它们转换为字符串键散列
hash.deep_transform_keys do |key|
key = "token" if key == :id
key.to_s
end
h.inject({}) { |m, (k,v)| m[k.sub(/^_/,'')] = v; m }
如果所有的键都是字符串,并且都有下划线前缀,那么你可以用这个来修补哈希:
h.keys.each { |k| h[k[1, k.length - 1]] = h[k]; h.delete(k) }
k[1, k.length - 1]位获取除第一个字符以外的所有k个字符。如果你想要一份拷贝,那么:
new_h = Hash[h.map { |k, v| [k[1, k.length - 1], v] }]
Or
new_h = h.inject({ }) { |x, (k,v)| x[k[1, k.length - 1]] = v; x }
如果你不喜欢用k[]符号来提取子字符串,你也可以使用sub:
h.keys.each { |k| h[k.sub(/\A_/, '')] = h[k]; h.delete(k) }
Hash[h.map { |k, v| [k.sub(/\A_/, ''), v] }]
h.inject({ }) { |x, (k,v)| x[k.sub(/\A_/, '')] = v; x }
并且,如果只有一些键有下划线前缀:
h.keys.each do |k|
if(k[0,1] == '_')
h[k[1, k.length - 1]] = h[k]
h.delete(k)
end
end
类似的修改可以对上述所有其他变体做,但这两个:
Hash[h.map { |k, v| [k.sub(/\A_/, ''), v] }]
h.inject({ }) { |x, (k,v)| x[k.sub(/\A_/, '')] = v; x }
如果没有额外的修改,应该可以使用没有下划线前缀的键。
以前的答案已经足够好了,但它们可能会更新原始数据。 如果您不希望原始数据受到影响,您可以试试我的代码。
newhash=hash.reject{|k| k=='_id'}.merge({id:hash['_id']})
首先,它将忽略键'_id',然后与更新后的键合并。
hash.each {|k,v| hash.delete(k) && hash[k[1..-1]]=v if k[0,1] == '_'}