ABC155のC問題を解いたので、思考プロセスを載せておきます

Contents

問題文

NN 枚の投票用紙があり、i (1≤i≤N)i (1≤i≤N) 枚目には文字列 SiSi が書かれています。書かれた回数が最も多い文字列を全て、辞書順で小さい順に出力してください。

ACしたコード

### SNIPPET
  # array = [*1..q].map { |_| gets.split.map(&:to_i) }
  # n = gets.split.map(&:to_i)
  # [].all?(&:even?)
  # a = [*1..m].repeated_combination(n).to_a
  # 切り捨て: .floor(2).to_f ,切り上げ: .ceil(2).to_f,四捨五入: round(2)
  # 3.upto(6) do |i|, 6.downto(3) do |i|
  # 公約数125.gcd(100)、公倍数125.lcm(100)
  # PI = Math::PI
  # 高さh = a * Math.sin(w / 180.0 * Math::PI)
  # 底辺 = a * Math.cos(w / 180.0 * Math::PI)
  def chmax(a, b) a > b ? a : b end
  # INF = Float::INFINITY
  # def chmin(a, b) a < b ? a : b end

n = gets.to_i
# a,b,c = gets.split.map(&:to_i)

s = [*1..n].map { |_| gets.chomp }
g = s.group_by(&:itself).map{|k,v| [k,v.count]}.to_h


max_count = 0
max_res = []
g.each do |k,v|
  if v > max_count
    max_res = [k]
    max_count = v
  elsif v == max_count
    max_res.push(k)
  end
end

max_res.sort.each do |ele|
  puts ele
end

キーポイント

まずは、それぞれの単語が何回出てきたかを計算します。

次に、多いものを抽出します。

今回は多かった単語は同着とみなし、どちらも表示なので、そこを悩みました。

思考プロセス

最初はrubyのメソッドでなんとかできるとわかりつつ、なかなかgroup_byが使い込めませんでした。

s.group_by(&:itself)がポイントで、

sが["aaa","bbb","aaa"]ならば、{"aaa"=>["aaa", "aaa"], "bbb"=>["bbb"]}こうなります。

理想は、{"aaa"=>2, "bbb"=>1}こんな感じなので、mapメソッドで、とりあえず、[["aaa", 2], ["bbb", 1]]これに直します。仕上げでto_hでハッシュにして出来上がり。

あとは、{"aaa"=>2, "bbb"=>1}から一番多かったものを計算し、表示しました。

投稿者 Ryuji_tech

インフラエンジニア→プログラミング講師→フロントエンジニア。スキル:HTML/CSS, Rails, React, Atcoder 茶 趣味:ワイン 人生最終目標:ワインとプログラミングを掛け合わせる。

コメントを残す

メールアドレスが公開されることはありません。 が付いている欄は必須項目です