0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

AtCoder ABC395 振り返り(緑コーダーがPythonでABCD問題)

Posted at

AtCoder ABC395を復習する

AtCoder Beginner Contest 395 - AtCoder

ABC394ABCDまでの4問解答でした。

B問題で思ったような結果が出なくて、時間がかかりました。

また、D問題も苦労して3回TLEを出してしまいました。
終了時間30秒前に「なんとかなれ〜(ハチワレ)」の勢いでダメ元の提出を実行したところ、ACになりました。
最近、時間制限ありの詰将棋アプリをやっていて、時間ギリギリまで粘って解答できた経験が良かったのかもしれません。

A - Strictly Increasing?

数列が単調増加かどうかを調べる問題。数列を初めから調べていき、一箇所でも単調増加でない場合はNoを出力して終了するようにしています。

N = int(input())
A = list(map(int, input().split()))
for i in range(1, N):
    if A[i-1] < A[i]:
        # 単調増加なら続行
        continue
    else:
        # 単調増加でない場合はNoを出力して終了
        print("No")
        exit()
print("Yes")

B - Make Target

これは凡ミスで解答に20分くらいかかってしまった。

ミスの内容は、

  • 問題文の座標系は1始まり、配列は0始まり
  • 描画する四角形の右下の座標を j まで描画するのですが、ループはj+1 までしないとダメ...

というのに躓いたのでした。

N = int(input())

answers = []
for i in range(N):
    s = ["_"] * N
    answers.append(s)

for i in range(1, N + 1):
    j = N + 1 - i
    if i > j:
        continue
    # 色を決定
    color = "#" if i % 2 == 1 else "."

    # 0始まりに合わせる
    correct_i = i - 1
    correct_j = j - 1 
    
    if j > N:
        continue

    # iからjの区間を塗る
    for mi in range(correct_i, correct_j + 1):
        for mj in range (correct_i, correct_j + 1):
            answers[mi][mj] = color

for i in range(N):
    line = answers[i]
    print("".join(line))

C - Shortest Duplicate Subarray

数列中で、最後に数値 x が出現した位置を、dict( last_index_dict ) に保存しておきます。
数値 x が再度出現した場合、その位置〜最後に出現した位置までが、対象の連続部分列となります。
連続部分列のサイズは その位置 - 最後に出現した位置 + 1 なので、最小値を更新していきます。

from collections import defaultdict

N = int(input())
A = list(map(int, input().split()))
last_index_dict = defaultdict(int) # 最後に数値が出現した位置を保存する
answer = INF

for i in range(N):
    a = A[i]
    if a in last_index_dict:
        # 数値がすでに出現している場合
        distance = i - last_index_dict[a] + 1

        # 最小値を更新
        answer = min(distance, answer)
        last_index_dict[a] = i
    else:
        # 数値が初めて出現する場合
        last_index_dict[a] = i

print(answer if answer != INF else -1)

D - Pigeon Swap

問題なのが操作2で、巣箱a内の鳩と、巣箱b内の鳩を入れ替えるという操作です。この操作をその通りに実装してしまうと、結構な数の鳩の移動操作が必要になり、TLEを出してしまいます。

そこで、この操作は巣箱をまるごと入れ替えることで、代用するように考えました。これにより、操作2の負荷を減らすことができ、TLEを回避することができます。

巣箱ごとの入れ替えを行うため、以下の変数を用意しました。

  • pointers は、巣箱の並び順を保存しておくリストです。pointers[1] は、1番目は 巣箱s かを保存しています。
    • pointersは、C言語のポインタのように、実際の巣箱への参照を保存するイメージです。
  • key_to_index は、巣箱s が何番目の巣箱かを保存しておく辞書です。key_to_index[s] は、巣箱s が何番目の巣箱かを保存しています。
from collections import defaultdict

N, Q = map(int, input().split())

positions = [] # 鳩n が、どの巣箱Sにいるか
pointers = [] # x 番目の巣箱は、どの巣箱かを保存
key_to_index = defaultdict(int) # 巣箱s は、x番目の巣箱かを保存

# 初期化
for i in range(N + 1):
    key = i
    positions.append(key) # 鳩1は巣箱1、鳩2は巣箱2、鳩3は巣箱3...
    pointers.append(key)  # 並びの1番目は巣箱1、2番めは巣箱2...
    key_to_index[key] = i # 巣箱1は1番目、巣箱2は2番目...

for _ in range(Q):
    query = list(map(int, input().split()))

    if query[0] == 1:
        # 鳩a を巣箱b に移動
        a, b = query[1], query[2]

        nest_a = positions[a]
        nest_b = pointers[b]

        if nest_a == nest_b:
            continue

        positions[a] = nest_b

    if query[0] == 2:
        # 巣箱a と巣箱b を入れ替える
        a, b = query[1], query[2]
        nest_a_key = pointers[a]
        nest_b_key = pointers[b]

        pointers[a], pointers[b] = nest_b_key, nest_a_key
        key_to_index[nest_a_key], key_to_index[nest_b_key] = b, a

    if query[0] == 3:
        # 鳩a は巣箱sにいる → 巣箱s は何番目の巣箱か を出力
        a = query[1]
        nest_key = positions[a]
        index = key_to_index[nest_key]
        print(index)
0
0
0

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
0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?