LoginSignup
1
0

More than 1 year has passed since last update.

概要

表題のプログラミング言語でAtCoder Beginners Selectionを実施記録した際の記録です
基本文法の学習が目的です

https://atcoder.jp/contests/abs
AtCoder Beginners Selectionとは?
このコンテストは、「AtCoderに登録したけど何をしていいか分からない・・・!」という人に向けて作られた、初心者向け問題集です。
問題は、dr.kenさんの、「AtCoder に登録したら次にやること ~ これだけ解けば十分闘える!過去問精選 10 問 ~」から選出しています。

基本的には以下ができれば大体の問題を解けるようになっている。

  • 標準入力
  • 標準出力
  • 四則演算
  • 条件分岐
  • 文字列操作
  • ループ
  • 配列の操作
  • 全通りの探索
  • 組み込み関数等の利用
  • ソート
  • 集合
  • 二次元配列の操作

PracticeA - Welcome to AtCoder

PracticeA - Welcome to AtCoder

問題文
高橋君はデータの加工が行いたいです。
整数 a,b,cと、文字列 s が与えられます。 a+b+c の計算結果と、文字列 s を並べて表示しなさい。

制約
1≤a,b,c≤1,000
1≤∣s∣≤100

入力
入力は以下の形式で与えられる。
a
b c
s

出力
a+b+c と s を空白区切りで 1 行に出力せよ。

基本的な標準入力、標準出力、四則演算ができればOK

main.sh
#!/bin/bash

# 整数 a, b, c を読み込む
read a
read b c

# a + b + c の結果を計算する
result=$((a + b + c))

# 文字列 s を読み込む
read s

# 結果と文字列 s を空白区切りで出力する
echo "$result $s"

ABC086A - Product

ABC086A - Product

問題文
シカのAtCoDeerくんは二つの正整数 a,b を見つけました。 a と b の積が偶数か奇数か判定してください。

制約
1 ≤ a,b ≤ 10000
a,b は整数

入力
入力は以下の形式で標準入力から与えられる。
a b

出力
積が奇数なら Odd と、 偶数なら Even と出力せよ。

基本的な標準入力、標準出力、四則演算、条件分岐ができればOK

main.sh
#!/bin/bash

# a と b の入力を受け取る
read a b

# a と b の積が偶数か奇数か判定する
if [ $((a * b % 2)) -eq 0 ]; then
  # a と b の積が 0 の場合は、積は 0 となるので、偶数とする
  echo "Even"
else
   echo "Odd"
fi

ABC081A - Placing Marbles

ABC081A - Placing Marbles

問題文
すぬけ君は 1,2,3 の番号がついた 3 つのマスからなるマス目を持っています。各マスには 0 か 1 が書かれており、マス i には si が書かれています。
すぬけ君は 1 が書かれたマスにビー玉を置きます。 ビー玉が置かれるマスがいくつあるか求めてください。

制約
s1 ,s2 ,s3 は 1 あるいは 0

入力
入力は以下の形式で標準入力から与えられる。
s1s2s3

出力
答えを出力せよ。

入力例 1
101

出力例 1
2

基本的な標準入力、標準出力、文字列操作ができればOK

main.sh
#!/bin/bash

# a と b の入力を受け取る
read s
# 入力された文字列をgrepコマンドに渡し、正規表現1にマッチする文字の個数を求めます。
echo "$s" | grep -o 1 | wc -l

ABC081B - Shift only

ABC081B - Shift only

問題文
黒板に N 個の正の整数 A1,...,AN が書かれています.
すぬけ君は,黒板に書かれている整数がすべて偶数であるとき,次の操作を行うことができます.
黒板に書かれている整数すべてを,2 で割ったものに置き換える.
すぬけ君は最大で何回操作を行うことができるかを求めてください.

制約
1≤N≤200
1≤Ai≤10^9

入力
入力は以下の形式で標準入力から与えられる。
N
A1 A2 ... AN

出力
すぬけ君は最大で何回操作を行うことができるかを出力せよ.

入力例 1
3
8 12 40

出力例 1
2

基本的な標準入力、標準出力、四則演算、ループ、条件分岐、配列の操作ができればOK

main.sh
#!/bin/bash

read n
read -a a
count_min=10000
for (( i = 0; i < n; i++ )); do
    count=0
    while (( $(( a[i] % 2 == 0 )) )); do
    	count=$(( count + 1 ))
        a[i]=$(( a[i] / 2 ))    
    done
    if (( $count < $count_min )); then
    	count_min=$count
    fi
done

# 結果を出力する
echo $count_min

ABC087B - Coins

ABC087B - Coins

問題文
あなたは、500 円玉を A 枚、100 円玉を B 枚、50 円玉を C 枚持っています。 これらの硬貨の中から何枚かを選び、合計金額をちょうど X 円にする方法は何通りありますか。
同じ種類の硬貨どうしは区別できません。2 通りの硬貨の選び方は、ある種類の硬貨についてその硬貨を選ぶ枚数が異なるとき区別されます。

制約
0≤A,B,C≤50
A+B+C≥1
50≤X≤20,000
A,B,C は整数である
X は 50 の倍数である

入力
入力は以下の形式で標準入力から与えられる。
A
B
C
X

出力
硬貨を選ぶ方法の個数を出力せよ。

基本的な標準入力、標準出力、四則演算、ループ、条件分岐、配列の操作、全通りの探索ができればOK

main.sh
#!/bin/bash

# 変数A,B,C,Xを受け取る
read A
read B
read C
read X

# 変数countを宣言し、0で初期化する
count=0

# 変数Aから変数Xまで、500円玉を1枚ずつ増やしながら繰り返す
for ((a=0; a<=$A; a++))
do
  # 変数Bから変数Xまで、100円玉を1枚ずつ増やしながら繰り返す
  for ((b=0; b<=$B; b++))
  do
    # 変数Cから変数Xまで、50円玉を1枚ずつ増やしながら繰り返す
    for ((c=0; c<=$C; c++))
    do
      # 変数sumに、500円玉の枚数×500 + 100円玉の枚数×100 + 50円玉の枚数×50を代入する
      sum=$((a*500 + b*100 + c*50))

      # 変数sumが変数Xと等しい場合
      if [ $sum -eq $X ]; then
        # 変数countに1を加える
        count=$((count+1))
      fi
    done
  done
done

# 変数countを出力する
echo $count

ABC083B - Some Sums

ABC083B - Some Sums

問題文
1 以上 N 以下の整数のうち、10 進法での各桁の和が A 以上 B 以下であるものの総和を求めてください。

制約
1≤N≤10^4
1≤A≤B≤36
入力はすべて整数である

入力
入力は以下の形式で標準入力から与えられる。
N A B

出力
1 以上 N 以下の整数のうち、10 進法での各桁の和が A 以上 B 以下であるものの総和を出力せよ。

基本的な標準入力、標準出力、四則演算、ループ、条件分岐、配列の操作、組み込み関数の利用、ができればOK

main.sh
#!/bin/bash

# 入力値を読み込む
read n a b

# 1からNまでの整数を順番に処理する
for ((i=1; i<=$n; i++))
do
  # i の各桁の和を求める
  sum=0
  for ((j=0; j<${#i}; j++))
  do
    sum=$((sum + ${i:$j:1}))
  done

  # sum が A 以上 B 以下である場合、総和を更新する
  if [[ $sum -ge $a && $sum -le $b ]]; then
    result=$((result + i))
  fi
done

# 総和を出力する
echo $result

ABC088B - Card Game for Two

ABC088B - Card Game for Two

問題文
N 枚のカードがあります. i 枚目のカードには, ai という数が書かれています.
Alice と Bob は, これらのカードを使ってゲームを行います. ゲームでは, Alice と Bob が交互に 1 枚ずつカードを取っていきます. Alice が先にカードを取ります.
2 人がすべてのカードを取ったときゲームは終了し, 取ったカードの数の合計がその人の得点になります. 2 人とも自分の得点を最大化するように最適な戦略を取った時, Alice は Bob より何点多く取るか求めてください.

制約
N は 1 以上 100 以下の整数
ai(1≤i≤N) は 1 以上 100 以下の整数

入力
入力は以下の形式で標準入力から与えられる.
N
a1 a2 a3 ... aN

出力
両者が最適な戦略を取った時, Alice は Bob より何点多く取るかを出力してください.

基本的な標準入力、標準出力、四則演算、条件分岐、配列の操作、ループ、ソート、ができればOK

main.sh
#!/bin/bash

read N
read -a cards

sorted_cards=($(printf "%s\n" "${cards[@]}" | sort -n -r | tr "\n" " "))

score=0
for ((i = 0; i < ${#sorted_cards[@]}; i++)); do
    if [ $((i % 2)) -eq 0 ]; then
        score=$((score+${sorted_cards[i]}))
    else
        score=$((score-${sorted_cards[i]}))
    fi
done
echo $score

ABC085B - Kagami Mochi

ABC085B - Kagami Mochi

問題文
X 段重ねの鏡餅 (X≥1) とは、X 枚の円形の餅を縦に積み重ねたものであって、どの餅もその真下の餅より直径が小さい(一番下の餅を除く)もののことです。例えば、直径 10、8、6 センチメートルの餅をこの順に下から積み重ねると 3 段重ねの鏡餅になり、餅を一枚だけ置くと 1 段重ねの鏡餅になります。
ダックスフンドのルンルンは N 枚の円形の餅を持っていて、そのうち i 枚目の餅の直径は diセンチメートルです。これらの餅のうち一部または全部を使って鏡餅を作るとき、最大で何段重ねの鏡餅を作ることができるでしょうか。

制約
1≤N≤100
1≤di≤100
入力値はすべて整数である。

入力
入力は以下の形式で標準入力から与えられる。
N
d1
​:
dN

​出力
作ることのできる鏡餅の最大の段数を出力せよ。

入力例 1
4
10
8
8
6

出力例 1
3

基本的な標準入力、標準出力、四則演算、ループ、配列、集合、ができればOK

main.sh
#!/bin/bash

read n
array=()
for ((i = 0; i < n; i++)); do
    read array[i]
done

sorted_array=($(printf "%s\n" "${array[@]}" | sort -n | tr "\n" " "))
count=1
for ((i = 1; i < n; i++)); do
    if [ ${sorted_array[i-1]} -lt ${sorted_array[i]} ]; then
        count=$((count+1))
    fi
done
echo $count

ABC085C - Otoshidama

ABC085C - Otoshidama

問題文
日本でよく使われる紙幣は、10000 円札、5000 円札、1000 円札です。以下、「お札」とはこれらのみを指します。
青橋くんが言うには、彼が祖父から受け取ったお年玉袋にはお札が N 枚入っていて、合計で Y 円だったそうですが、嘘かもしれません。このような状況がありうるか判定し、ありうる場合はお年玉袋の中身の候補を一つ見つけてください。なお、彼の祖父は十分裕福であり、お年玉袋は十分大きかったものとします。

制約
1≤N≤2000
1000≤Y≤2×10^7
N は整数である。
Y は 1000 の倍数である。

入力
入力は以下の形式で標準入力から与えられる。
N Y

出力
N 枚のお札の合計金額が Y 円となることがありえない場合は、-1 -1 -1 と出力せよ。
N 枚のお札の合計金額が Y 円となることがありうる場合は、そのような N 枚のお札の組み合わせの一例を「10000 円札 x 枚、5000 円札 y 枚、1000 円札 z 枚」として、x、y、z を空白で区切って出力せよ。複数の可能性が考えられるときは、そのうちどれを出力してもよい。

入力例 1
9 45000

出力例 1
4 0 5

基本的な標準入力、標準出力、四則演算、ループができればOK

※ 以下はTLEになり確認中、どなたかアドバイス頂けると助かります..

main.sh
#!/bin/bash

# 入力値を変数に格納
read n Y

# 入力値から、10000円札、5000円札、1000円札の枚数を求める
for ((i=0; i<=n; i++)); do
  for ((j=0; j<=n-i; j++)); do

    # 10000円札の枚数を求める
    x=$i
    # 5000円札の枚数を求める
    y=$j
    # 1000円札
    k=$((n-i-j))

    # 10000円札の枚数 × 10000 + 5000円札の枚数 × 5000 + 1000円札の枚数 × 1000が
    # y円になる場合、答えを出力して終了
    if [ $((x*10000 + y*5000 + k*1000)) -eq $Y ]; then
      echo "$x $y $k"
      exit 0
    fi
    done
done
# 見つからない場合は-1 -1 -1を出力
echo "-1 -1 -1"

ABC049C - 白昼夢

ABC049C - 白昼夢

問題文
英小文字からなる文字列 S が与えられます。 Tが空文字列である状態から始め、以下の操作を好きな回数繰り返すことで S=T とすることができるか判定してください。

制約
1≦∣S∣≦10^5
S は英小文字からなる。

入力
入力は以下の形式で標準入力から与えられる。
S

出力
S=T とすることができる場合 YES を、そうでない場合 NO を出力せよ。

入力例 1
erasedream

出力例 1
YES

基本的な標準入力、標準出力、文字列操作ができればOK

※ 以下はTLEになり確認中、どなたかアドバイス頂けると助かります..

main.sh
#!/bin/bash

read S
while [ -n "$S" ]; do
    if [[ $S =~ dream$ ]]; then
        # 文字列Sから、dreamを末尾から削除する
        S=${S%dream}
        continue
    elif [[ $S =~ dreamer$ ]]; then
        # 文字列Sから、dreamerを末尾から削除する
        S=${S%dreamer}
        continue
    elif [[ $S =~ erase$ ]]; then
        # 文字列Sから、eraseを末尾から削除する
        S=${S%erase}
        continue
    elif [[ $S =~ eraser$ ]]; then
        # 文字列Sから、eraserを末尾から削除する
        S=${S%eraser}
        continue
    fi
    echo "NO"
    exit
done

# 文字列Sが空文字列の場合に、YESを出力する
echo "YES"

ABC086C - Traveling

ABC086C - Traveling

問題文
シカのAtCoDeerくんは二次元平面上で旅行をしようとしています。 AtCoDeerくんの旅行プランでは、時刻 0 に 点 (0,0) を出発し、 1 以上 N 以下の各 i に対し、時刻tiに 点 (xi,yi) を訪れる予定です。
AtCoDeerくんが時刻 t に 点 (x,y) にいる時、 時刻 t+1 には 点 (x+1,y), (x−1,y), (x,y+1), (x,y−1) のうちいずれかに存在することができます。 その場にとどまることは出来ないことに注意してください。 AtCoDeerくんの旅行プランが実行可能かどうか判定してください。

制約
1 ≤ N ≤ 10^5
0 ≤ xi ≤ 10^5
0 ≤ yi ≤ 10^5
1 ≤ ti ≤ 10^5
ti < ti+1 (1 ≤ i ≤ N−1)
入力は全て整数

入力
入力は以下の形式で標準入力から与えられる。
N
t1 x1 y1
​t2 x2 y2
:
tN xN yN

出力
旅行プランが実行可能ならYesを、不可能ならNoを出力してください。

入力例 1
2
3 1 2
6 1 1

出力例 1
Yes

入力例 2
1
2 100 100

出力例 2
No
(0,0) にいる状態から 2 秒後に (100,100) にいるのは不可能です。

入力例 3
2
5 1 1
100 1 1

出力例 3
No

基本的な標準入力、標準出力、四則演算、ループ、二次元配列の操作、ができればOK

※ 以下はTLEになり確認中、どなたかアドバイス頂けると助かります..

main.sh
#!/bin/bash

# 入力を受け取る
read N

# 前回の座標を保持する変数を定義する
previous_time=0
previous_x=0
previous_y=0

# N回繰り返す
for i in `seq 1 $N`
do
  # 時刻 ti, xi, yi を受け取る
  read ti xi yi

  # 現在の座標から前回の座標へ移動するために必要な時間を求める
  # 現在の座標から前回の座標への移動距離を求める
  dx=$((xi - previous_x))
  dy=$((yi - previous_y)) 
  check=$((ti - previous_time - ${dx#-} - ${dy#-}))
  # 移動距離が時間以上であれば旅行プランは実行不可能
  if  [[ ${check} -lt 0 ]] || [[ $((check % 2)) -ne 0 ]]; then
    echo "No"
    exit
  fi

  # 現在の座標を前回の座標として保存する
  previous_time=$ti
  previous_x=$xi
  previous_y=$yi
done

# すべての移動について確認した結果、実行可能な旅行プランであればYesを出力する
echo "Yes"

参考

1
0
2

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
1
0