ハッシュの配列を欠落値で埋めます

Nov 25 2020

ApexChartのヒートマップ固有のグラフを作成しようとしています。これまでのところ、3つのアクティビティでこのハッシュの配列を取得しました。

[{
  :name => "Activity 1",
  :data => {
    "May 2020" => 37, "June 2020" => 17, "July 2020" => 9, "August 2020" => 18
  }
}, {
  :name => "Activity 2",
  :data => {
    "May 2020" => 3
  }
}, {
  :name => "Activity 3",
  :data => {
    "July 2020" => 5, "November 2020" => 11
  }
}]

活動3で、我々はわずか2ヶ月になった、JulyNovember

私のニーズは、ハッシュごとに、欠落しているすべての日付を入力し、値として0を入力することです。私の待望の結果は

[{
  :name => "Activity 1",
  :data => {
    "May 2020" => 37, "June 2020" => 17, "July 2020" => 9, "August 2020" => 18, "November 2020" => 0
  }
}, {
  :name => "Activity 2",
  :data => {
    "May 2020" => 3, "June 2020" => 0, "July 2020" => 0, "August 2020" => 0, "November 2020" => 0
  }
}, {
  :name => "Activity 3",
  :data => {
    "May 2020" => , "June 2020" => 0, "July 2020" => 5, "August 2020" => 0, "November 2020" => 11
  }
}]

はい、Septemberわざと行方不明です。これを達成するための最良の方法は、毎月1つずつ取得することだと思います。次に、各配列を欠落している月で埋めます。しかし、私はこれを達成する方法がわかりません。

回答

1 CarySwoveland Nov 26 2020 at 02:57

arrがハッシュの配列である場合、2つのステップで目的の配列を作成できます。

require 'date'
date_fmt = "%B %Y"
first_month, last_month = arr.flat_map do |g|
  g[:data].keys
end.map { |s| Date.strptime(s, date_fmt) }.minmax
  #=> [#<Date: 2020-05-01 ((2458971j,0s,0n),+0s,2299161j)>,
  #    #<Date: 2020-11-01 ((2459155j,0s,0n),+0s,2299161j)>] 

h = (first_month..last_month).map do |d|
  d.strftime(date_fmt)
end.product([0]).to_h
  #=> {"May 2020"=>0, "June 2020"=>0, "July 2020"=>0, "August 2020"=>0,
  #    "September 2020"=>0, "October 2020"=>0, "November 2020"=>0} 
arr.map { |g| g.merge(:data => h.merge(g[:data])) }
  #=> [
  #     {
  #       :name=>"Activity 1",
  #       :data=>{
  #         "May 2020"=>37, "June 2020"=>17, "July 2020"=>9,
  #         "August 2020"=>18, "September 2020"=>0,
  #         "October 2020"=>0, "November 2020"=>0
  #       }
  #     },
  #     {
  #       :name=>"Activity 2",
  #       :data=>{
  #         "May 2020"=>3, "June 2020"=>0, "July 2020"=>0,
  #         "August 2020"=>0, "September 2020"=>0,
  #         "October 2020"=>0, "November 2020"=>0
  #       }
  #     },
  #     {
  #       :name=>"Activity 3",
  #       :data=>{
  #         "May 2020"=>0, "June 2020"=>0, "July 2020"=>5,
  #         "August 2020"=>0, "September 2020"=>0,
  #         "October 2020"=>0, "November 2020"=>11
  #       }
  #     }
  #   ] 

参照してください可算#のflat_map、日付::のstrptime、アレイ#1 MINMAX、日付#1のstrftime、アレイ#製品とハッシュ#マージを。日付フォーマットディレクティブについては、DateTime#strptimeも参照してください。

注の計算でいるfirst_monthlast_month

[#<Date: 2020-05-01 ((2458971j,0s,0n),+0s,2299161j)>,
 #<Date: 2020-11-01 ((2459155j,0s,0n),+0s,2299161j)>].
  map { |d| d.strftime(date_fmt) }
  #=> ["May 2020", "November 2020"]
1 KARASZIIstván Nov 25 2020 at 18:27

これは、最初に月をフェッチしてから、ケースのハッシュである配列アイテムを変更することで、同様の方法で実現できます。

months = data.flat_map { |d| d[:data].keys }.to_set
data.each do |d|
  months.each do |month|
    d[:data][month] = 0 unless d[:data].key?(month)
  end
end

またArrayHash値がデフォルト値で初期化される新しいものを作成することもできます。

data_with_default = data.map do |d|
  {
    name: d[:name],
    data: Hash.new(0).update(d[:data])
  }
end
BenTrewern Nov 25 2020 at 18:48

次のことを試してください。

a = [{
  :name => "Activity 1",
  :data => {
    "May 2020" => 37, "June 2020" => 17, "July 2020" => 9, "August 2020" => 18
  }
}, {
  :name => "Activity 2",
  :data => {
    "May 2020" => 3
  }
}, {
  :name => "Activity 3",
  :data => {
    "July 2020" => 5, "November 2020" => 11
  }
}]

months = ['May 2020', 'June 2020', 'July 2020', 'August 2020', 'November 2020']

puts a.map { |h| { name: h[:name], data: Hash[months.map { |m| [m, h[:data][m] ? h[:data][m] : 0] }] } }