Преобразование глубокого вложенного хэша в многомерный массив в 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]]
Другие вопросы по тегам