繁体   English   中英

Ruby遍历哈希哈希

[英]Ruby iterating over hash of hash

我有以下数组,并且正在努力格式化以满足我的需求。

consolidated = [
  {:name=>"Bob",   :details=>{"work"=>"Carpenter", "age"=>"26", "Experience"=>"6"} },
  {:name=>"Colin", :details=>{"work"=>"painting",  "age"=>"20", "Experience"=>"4"} }
]

我正在尝试将其格式设置如下:

Bob    work         Carpenter
       age          26
       Experience   6

Colin  work         painting
       age          20
       Experience   4

我尝试了以下方法:

require 'csv'
CSV.open("output.csv", "wb") do |csv|
  csv << ["name", "nature", "details"]
  consolidated.each do |val|
    csv << [val[:name], val[:details]]
  end
end
  #=> [{:name=>"Bob",   :details=>{"work"=>"Carpenter", "age"=>"26", "Experience"=>"6"}},
  #    {:name=>"Colin", :details=>{"work"=>"painting", "age"=>"20", "Experience"=>"4"}}]

但它打印以下内容

name      nature                                                  details

Bob      "work"=>"Carpenter", "age"=>"26", "Experience"=>"6"

Colin    "work"=>"painting", "age"=>"20", "Experience"=>"4"

我不完全确定如何从第一个循环中迭代散列的散列,只得到预期的格式。

谢谢。

以下是一些入门知识:

require 'csv'
data = [
  {:name => "Bob", :details=>{"work"=>"Carpenter", "age"=>"26", "Experience"=>"6"}},
  {:name => "Colin", :details=>{"work"=>"painting", "age"=>"20", "Experience"=>"4"}}
]

str = CSV.generate do |csv|
  data.each do |datum|
    datum[:details].each do |detail_key, detail_value|
      csv << [datum[:name], detail_key, detail_value]
    end
  end
end

puts str
# >> Bob,work,Carpenter
# >> Bob,age,26
# >> Bob,Experience,6
# >> Colin,work,painting
# >> Colin,age,20
# >> Colin,Experience,4

只需迭代所有细节,并为那里的每个键值对发出新行,并添加一个人的名字。

这将为您提供几乎所需的东西。 在各行和个人姓名之间仅缺少空白行在每行中重复。 找出如何添加这些改进将是您的家庭作业。

我不知道CSV生成(因此,假设它按您编写的方式工作),您可以通过以下方式迭代对象:

consolidated = [{:name => "Bob", :details=>{"work"=>"Carpenter", "age"=>"26", "Experience"=>"6"}}, {:name => "Colin", :details=> {"work"=>"painting", "age"=>"20", "Experience"=>"4"}}]


CSV.open("output.csv", "wb") do |csv|
  csv << ["name", "nature", "details"]

  consolidated.each do |val|
    details = val[:details]
    nature_1 = details.keys.first
    detail_1 = details.delete(nature_1)
    csv << [val[:name], nature_1, detail_1]

    details.each do |k, v|
      csv << [nil, k, v]
    end
  end
end

注意:这会损坏您consolidated的原始数据数组。 因此,如果要保留它,请先将其dup 或修改逻辑以不从val[:details]删除第一个键值。

您需要通过each_pair迭代器迭代嵌入式哈希。 像这样:

data = {:name => "Bob", :details=>{"work"=>"Carpenter", "age"=>"26", "Experience"=>"6"}}

CSV.open("output.csv", "wb") do |csv|

  csv << ["name", "nature", "details"]

  data.each do |val|

    csv << [ val[:name], val[:details]['work'] ]

    data[:details].each_pair do |key, value]

      # here we have to drop the first pair because i've used it earlier 
      next if key == 'work'

      csv << [ "", key, value ]

    end

  end

end

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM