Zenn
🤖

ABC231 A〜D をPython で解く

2021/12/11に公開

ABC231 A〜D Pythonで解く

パナソニックプログラミングコンテスト2021(AtCoder Beginner Contest 231)を Python で解いたので、解説します。

A - Water Pressure

  • 最近 A 問題難しめなので身構えてしまいますが、100で割るだけです。
d = int(input())
print(d / 100)

B - Election

  • 辞書を使って集計しました。

  • value 最大のキーを取り出す方法がわからず、自分で書くか迷って結局ググりました。こういうロスがよくない。

n = int(input())
dict_ = dict()
for _ in range(n):
    s = input()
    if s in dict_.keys():
        dict_[s] += 1
    else:
        dict_[s] = 1

print(max(dict_, key=dict_.get)) 

C - Counting 2

  • xjx_j 以上の要素の個数を全探索で数えることができますが、これは O(N)O(N)で、全体で O(QN)O(QN) となり、間に合いません。

  • クエリがたくさんくる場合は、各クエリを高速で処理する必要があります。配列の順序は関係ないので、ソートしてから二分探索で処理しましょう。すると各クエリを O(logN)O(\log N) で計算でき、全体で O(NlogN)O(N\log N) の計算量で済みます。

  • Python には bisect という標準ライブラリが使えます。ここでは、二分探索を自分で実装しています。

n, q = map(int, input().split())
A = list(map(int, input().split()))
A.sort()
ans = []
for _ in range(q):
    x = int(input())
    if A[-1] < x:
        ans.append(0)
        continue
    ok = n - 1
    ng = -1
    while abs(ok - ng) > 1:
        mid = (ok + ng) // 2
        if A[mid] >= x:
            ok = mid
        else:
            ng = mid
    ans.append(n - ok)

for i in range(q):
    print(ans[i])

D - Neighbors

  • 想定とは違うかもしれないので、あとで解説を確認します。

  • 2500 人も通していて、すごい...。

  • 順番に条件に出てくる a と b を隣り合うようにならべていくことを考えればよいですが、すでに両隣が埋まっている、または制約に従うとループになってしまう場合を避ける必要があります。 Union Find というデータ構造を使い実装しました。ただ、Union Find だけでは解けないのでグラフも用意して隣になにが来ているかを管理しました。

  • a と b を Union Find で合併していきますが、同じグループで隣ではない場合、ループになることがわかります。

# Union Find の実装部分は整理できていないため、ここでは省略します
n, m = map(int, input().split())
G = [[] for _ in range(n+1)]
uf = UnionFind(n+1)
for i in range(m):
    a, b = map(int, input().split())
    if (b in G[a]) or (a in G[b]):
        continue
    if max(len(G[a]), len(G[b])) >= 2:
        print('No')
        exit()
    if uf.same(a, b):
        print('No')
        exit()
    uf.union(a, b)
    G[a].append(b)
    G[b].append(a)
    
print('Yes')

おつかれさまでした。

Discussion

ログインするとコメントできます