我正在尝试创建一个输出JSON的所有条目的键和值的函数,如下所述,因此我可以使用它发送类似于以下内容的信息:
key = "id", value = 1
key = "mem/stat1", value = 10
key = "more_stats/extra_stats/stat7", value = 5
示例JSON:
my_json =
{
"id": 1,
"system_name": "System_1",
"mem" : {
"stat1" : 10,
"stat2" : 1056,
"stat3" : 10563,
},
"other_stats" : {
"stat4" : 1,
"stat5" : 2,
"stat6" : 3,
},
"more_stats" : {
"name" : "jdlfjsdlfjs",
"os" : "fjsalfjsl",
"error_count": 3
"extra_stats" : {
"stat7" : 5,
"stat8" : 6,
},
}
}
我从这个问题(need help in getting nested ruby hash hierarchy)中找到了一个有用的答案,但是即使进行了一些更改,它仍然无法满足我的要求:
def hashkeys(json, keys = [], result = [])
if json.is_a?(Hash)
json.each do |key, value|
hashkeys(value, keys + [key], result)
end
else
result << keys
end
result.join("/")
end
它将所有键一起作为一个字符串返回,并且没有正确包含任何相应的值。
当前不需要的哈希键输出:
id/system_name/mem/stat1/mem/stat2/...
理想情况下,我想要包含my_json
的东西:
find_nested_key_value(my_json)
some logic loop involving key and value:
if more logic needed
another_logic_loop_for_more_nested_info
else
send_info(final_key, final_value)
end
end
end
因此,如果final_key = "mem/stat1"
然后final_value = 10
,那么下一个迭代将是final_key = "mem/stat2"
和final_value = 1056
,依此类推
我该如何实现?并且正在使用hashkeys
之类的函数来实现此目标的最佳方法
答案 0 :(得分:2)
这是一种递归方法,它将创建“扁平哈希”,即没有嵌套的哈希,其中的键是由斜杠分隔的嵌套键。
def flatten_hash(hash, result = {}, prefix = nil)
hash.each do |k,v|
if v.is_a? Hash
flatten_hash(v, result, [prefix, k].compact.join('/'))
else
result[[prefix, k].compact.join('/')] = v
end
end
result
end
my_hash = {'id': 1, 'system_name': 'Sysem_1', 'mem': {'stat1': 10, 'stat2': 1056, 'stat3': 10563}}
flatten_hash(my_hash)
=> {"id"=>1, "system_name"=>"Sysem_1", "mem/stat1"=>10, "mem/stat2"=>1056, "mem/stat3"=>10563}
答案 1 :(得分:0)
def key_path_value(key_string, my_json)
value = nil
key_array = key_string.split("/")
return value if key_array.empty?
return my_json[key_array.last] if key_array.length == 1
value = my_json[key_array.first.to_sym]
key_array = key_array.drop(1)
key_array.each do |key|
break unless value.is_a? Hash
value = value[key.to_sym]
end
return value
end