У меня есть вложенный хэш, и мне нужен один массив, где каждый элемент представляет собой массив ключей, представляющий путь через вложенный хэш к не-хэшам (листовые узлы).
Например, с учетом ввода:
x = Hash.new
x["a"] = Hash.new
x["a"]["b"] = Hash.new
x["a"]["b"]["c"] = "one"
x["a"]["b"]["d"] = "two"
x["a"]["e"] = "three"
x["f"] = Hash.new
x["f"]["g"] = "four"
Я хотел бы вывод:
[["a", "b", "c"], ["a", "b", "d"], ["a", "e"], ["f", "g"]]
Приведенный ниже код работает с использованием двух рекурсивных методов: один для генерации вложенного массива, а другой для его удаления!.
Моя интуиция Ruby говорит мне, что должен быть гораздо более эффективный и элегантный способ достижения этого. Кто-нибудь может предложить решение для игры в гольф или решение Ruby Way?
def collect_key_paths(object, path=[])
result = nil
if object.is_a?(Hash)
path_for_current_hash = object.map do |key, value|
incremented_path = [path, key].flatten
collect_key_paths(value, incremented_path)
end
result = path_for_current_hash
else
result = path
end
result
end
def smoothe(array, store=[])
if array.none? { |element| element.is_a?(Array) }
store << array
else
array.each do |element|
store = smoothe(element, store)
end
end
store
end
x = Hash.new
x["a"] = Hash.new
x["a"]["b"] = Hash.new
x["a"]["b"]["c"] = "one"
x["a"]["b"]["d"] = "two"
x["a"]["e"] = "three"
x["f"] = Hash.new
x["f"]["g"] = "four"
nested_key_paths = collect_key_paths(x)
puts "RESULT:#{smoothe(nested_key_paths)}"
Результат этого кода, работающего в версии 1.9.2:
RESULT:[["a", "b", "c"], ["a", "b", "d"], ["a", "e"], ["f", "g"]]
Спасибо!