h = {
data: {
user: {
value: "John Doe"
}
}
}
要为嵌套哈希值赋值,我们可以使用
h[:data][:user][:value] = "Bob"
但是,如果缺少中间的任何部分,则会导致错误。
像
这样的东西h.dig(:data, :user, :value) = "Bob"
不起作用,因为还没有Hash#dig=
可用。
为了安全地分配价值,我们可以做到
h.dig(:data, :user)&.[]=(:value, "Bob") # or equivalently
h.dig(:data, :user)&.store(:value, "Bob")
但有更好的方法吗?
答案 0 :(得分:12)
这不是没有它的警告(如果你从其他地方收到哈希值,那就不行了),但是一个常见的解决方案就是:
hash = Hash.new {|h,k| h[k] = h.class.new(&h.default_proc) }
hash[:data][:user][:value] = "Bob"
p hash
# => { :data => { :user => { :value => "Bob" } } }
答案 1 :(得分:5)
以@rellampec的答案为基础,不会引发错误:
def dig_set(obj, keys, value)
key = keys.first
if keys.length == 1
obj[key] = value
else
obj[key] = {} unless obj[key]
dig_set(obj[key], keys.slice(1..-1), value)
end
end
obj = {d: 'hey'}
dig_set(obj, [:a, :b, :c], 'val')
obj #=> {d: 'hey', a: {b: {c: 'val'}}}
答案 2 :(得分:0)
我找到了一种简单的解决方案,即使缺少父键(即使哈希已经存在),也可以设置嵌套哈希的值。鉴于:
x = { gojira: { guitar: { joe: 'charvel' } } }
假设您要包括马里奥的鼓来产生:
x = { gojira: { guitar: { joe: 'charvel' }, drum: { mario: 'tama' } } }
我结束了猴子补丁哈希:
class Hash
# ensures nested hash from keys, and sets final key to value
# keys: Array of Symbol|String
# value: any
def nested_set(keys, value)
raise "DEBUG: nested_set keys must be an Array" unless keys.is_a?(Array)
final_key = keys.pop
return unless valid_key?(final_key)
position = self
for key in keys
return unless valid_key?(key)
position[key] = {} unless position[key].is_a?(Hash)
position = position[key]
end
position[final_key] = value
end
private
# returns true if key is valid
def valid_key?(key)
return true if key.is_a?(Symbol) || key.is_a?(String)
raise "DEBUG: nested_set invalid key: #{key} (#{key.class})"
end
end
用法:
x.nested_set([:instrument, :drum, :mario], 'tama')
您的示例的用法:
h.nested_set([:data, :user, :value], 'Bob')
我错过了任何警告吗?在不牺牲可读性的情况下编写代码的更好方法?
答案 3 :(得分:0)
有趣的一个:
def dig_set(obj, keys, value)
if keys.length == 1
obj[keys.first] = value
else
dig_set(obj[keys.first], keys.slice(1..-1), value)
end
end
如果没有[]
或[]=
方法,总会引发异常。
答案 4 :(得分:0)
在寻找类似问题的答案时,我在发展过程中偶然发现了一个类似于 @niels-kristian 答案的界面,但还想支持命名空间定义参数,如 xpath。
def deep_merge(memo, source)
# From: http://www.ruby-forum.com/topic/142809
# Author: Stefan Rusterholz
merger = proc { |key, v1, v2| Hash === v1 && Hash === v2 ? v1.merge(v2, &merger) : v2 }
memo.merge!(source, &merger)
end
# Like Hash#dig, but for setting a value at an xpath
def bury(memo, xpath, value, delimiter=%r{\.})
xpath = xpath.split(delimiter) if xpath.respond_to?(:split)
xpath.map!{|x|x.to_s.to_sym}.push(value)
deep_merge(memo, xpath.reverse.inject { |memo, field| {field.to_sym => memo} })
end
嵌套哈希有点像 xpath,而 dig
的反义词是 bury
。
irb(main):014:0> memo = {:test=>"value"}
=> {:test=>"value"}
irb(main):015:0> bury(memo, 'test.this.long.path', 'value')
=> {:test=>{:this=>{:long=>{:path=>"value"}}}}
irb(main):016:0> bury(memo, [:test, 'this', 2, 4.0], 'value')
=> {:test=>{:this=>{:long=>{:path=>"value"}, :"2"=>{:"4.0"=>"value"}}}}
irb(main):017:0> bury(memo, 'test.this.long.path.even.longer', 'value')
=> {:test=>{:this=>{:long=>{:path=>{:even=>{:longer=>"value"}}}, :"2"=>{:"4.0"=>"value"}}}}
irb(main):018:0> bury(memo, 'test.this.long.other.even.longer', 'other')
=> {:test=>{:this=>{:long=>{:path=>{:even=>{:longer=>"value"}}, :other=>{:even=>{:longer=>"other"}}}, :"2"=>{:"4.0"=>"value"}}}}