Преобразование глубокого вложенного хэша в многомерный массив в ruby
Я хочу превратить вложенные хэши в рекурсивные массивы многомерных.
Это мой текущий код:
def deep_to_a(hash)
return hash.to_a.each {|k, v|
if k.is_a?(Hash) then
hash[k.to_a] = hash.delete(k)
deep_to_a(k)
elsif v.is_a?(Hash) then
hash[k] = hash[k].to_a
if hash[k].any?{|k, v| k.is_a?(Hash) || v.is_a?(Hash)}
deep_to_a(v)
end
end
}
end
Я хочу получить:
h = {11=>{12=>13, 14=>15}, 16=>17}
p deep_to_a(h) # => [[11, [[12, 13], [14, 15]]], [16, 17]]
Но я получаю
[[11, {12=>13, 14=>15}], [16, 17]]
Как я могу заставить это работать?
2 ответа
Решение
Деструктивную функцию трудно отладить. В этом случае, map
лучше, чем each
+ разрушительное назначение.
Результат hash.to_a
это массив, так что ваша итерация |k, v|
это неверно.
def deep_to_a(hash)
hash.map do |v|
if v.is_a?(Hash) or v.is_a?(Array) then
deep_to_a(v)
else
v
end
end
end
h = {11=>{12=>13, 14=>15}, 16=>17}
p deep_to_a(h)
# [[11, [[12, 13], [14, 15]]], [16, 17]]
def deep_to_a(h)
h.map { |k,v| [k, Hash === v ? deep_to_a(v) : v] }
end
deep_to_a({11=>{12=>13, 14=>15}, 16=>17})
#=> [[11, [[12, 13], [14, 15]]], [16, 17]]