用ruby从数组创建嵌套散列(creating nested hash from array with ruby)
我试图重新排列数组中的数据,以散列格式,但我相当混乱使用嵌套if
样本数据
[ ["England", "London", "University College London ", "Faculty of Law"], ["England", "London", "University College London ", "Faculty of Engineering"], ["England", "London", "Imperial College London", "Faculty of Medicine"], ["England", "Manchester", "University of Manchester", "Faculty of Engineering"] ]预期产出
{:name=>"England", :level=>1, :children=>[{:name=>"London", :level=>2, :children=>[{:name=>"University College London ", :level=>3, :children=>[{:name=>"Faculty of Law", :level=>4, :children=>[]}, {:name=>"Faculty of Engineering", :level=>4, :children=>[]}]}, {:name=>"Imperial College London", :level=>3, :children=>[{:name=>"Faculty of Engineering", :level=>4, :children=>[]}] }] }] }希望我已经提供了明确的解释
PS。 编辑,以显示我已经尝试过的第一次,我做了散列数组,然后做这样的事情,我认为它不会混淆这么多
result = [] arr.each do |b| if result.any? {|r| r[:name] == b[:name]} if result.first[:children].any? {|r| r[:name] == b[:children].first[:name]} if result.first[:children].any?{|c| c[:children].any? {|r| r[:name] == b[:children].first[:children].first[:name] && c[:name] == b[:children].first[:name] }} if result.first[:children].any? {|r| r[:children].any? {|c| c[:children].any?{|k| k[:name] == b[:children].first[:children].first[:children].first[:name] && (c[:name] == b[:children].first[:children].first)}}} else result.first[:children].any?{|c| c[:children].any? {|r| r[:name] == b[:children].first[:children].first[:name] ; r[:children] << b[:children].first[:children].first[:children].first}} end #fourth else result.first[:children].any? {|r| r[:name] == b[:children].first[:name]; r[:children] << b[:children].first[:children].first} end else result.any? {|r| r[:name] == b[:name] ; r[:children] << b[:children].first} end else result << b end endI'm trying to rearrange data in array to be in hash format but I pretty messed up using nested if
Sample Data
[ ["England", "London", "University College London ", "Faculty of Law"], ["England", "London", "University College London ", "Faculty of Engineering"], ["England", "London", "Imperial College London", "Faculty of Medicine"], ["England", "Manchester", "University of Manchester", "Faculty of Engineering"] ]Expected Output
{:name=>"England", :level=>1, :children=>[{:name=>"London", :level=>2, :children=>[{:name=>"University College London ", :level=>3, :children=>[{:name=>"Faculty of Law", :level=>4, :children=>[]}, {:name=>"Faculty of Engineering", :level=>4, :children=>[]}]}, {:name=>"Imperial College London", :level=>3, :children=>[{:name=>"Faculty of Engineering", :level=>4, :children=>[]}] }] }] }hope I have provide clear explaination
ps. edit to show what I've tried first i make array of hash then do something like this I thought it wouldn't be confused this much
result = [] arr.each do |b| if result.any? {|r| r[:name] == b[:name]} if result.first[:children].any? {|r| r[:name] == b[:children].first[:name]} if result.first[:children].any?{|c| c[:children].any? {|r| r[:name] == b[:children].first[:children].first[:name] && c[:name] == b[:children].first[:name] }} if result.first[:children].any? {|r| r[:children].any? {|c| c[:children].any?{|k| k[:name] == b[:children].first[:children].first[:children].first[:name] && (c[:name] == b[:children].first[:children].first)}}} else result.first[:children].any?{|c| c[:children].any? {|r| r[:name] == b[:children].first[:children].first[:name] ; r[:children] << b[:children].first[:children].first[:children].first}} end #fourth else result.first[:children].any? {|r| r[:name] == b[:children].first[:name]; r[:children] << b[:children].first[:children].first} end else result.any? {|r| r[:name] == b[:name] ; r[:children] << b[:children].first} end else result << b end end最满意答案
你可以这样做递归:
def map_objects(array, level = 1) new_obj = [] array.group_by(&:shift).each do |key, val| new_obj << {:name=>key, :level=>level, :children=>map_objects(val, level + 1)} if key end new_obj end对于你的数组,它会像这样返回:
# [ # {:name => "England", :level => 1, :children => [ # {:name => "London", :level => 2, :children => [ # {:name => "University College London ", :level => 3, :children => [ # {:name => "Faculty of Law", :level => 4, :children => [] # }, # {:name => "Faculty of Engineering", :level => 4, :children => [] # }] # }, # {:name => "Imperial College London", :level => 3, :children => [ # {:name => "Faculty of Medicine", :level => 4, :children => [] # }] # }] # }, # {:name => "Manchester", :level => 2, :children => [ # {:name => "University of Manchester", :level => 3, :children => [ # {:name => "Faculty of Engineering", :level => 4, :children => [] # }] # }] # }] # } # ]You could do this recursive like this:
def map_objects(array, level = 1) new_obj = [] array.group_by(&:shift).each do |key, val| new_obj << {:name=>key, :level=>level, :children=>map_objects(val, level + 1)} if key end new_obj endFor your array it will return like this:
# [ # {:name => "England", :level => 1, :children => [ # {:name => "London", :level => 2, :children => [ # {:name => "University College London ", :level => 3, :children => [ # {:name => "Faculty of Law", :level => 4, :children => [] # }, # {:name => "Faculty of Engineering", :level => 4, :children => [] # }] # }, # {:name => "Imperial College London", :level => 3, :children => [ # {:name => "Faculty of Medicine", :level => 4, :children => [] # }] # }] # }, # {:name => "Manchester", :level => 2, :children => [ # {:name => "University of Manchester", :level => 3, :children => [ # {:name => "Faculty of Engineering", :level => 4, :children => [] # }] # }] # }] # } # ]更多推荐
发布评论