LoginSignup
1
1

More than 3 years have passed since last update.

AtCoder Beginner Contest 189 参戦記

Last updated at Posted at 2021-01-23

AtCoder Beginner Contest 189 参戦記

終了50秒前に書けたE問題を投げたら奇跡的に一発 AC が出て久々に大勝利!

ABC189A - Slot

1分半で突破. 書くだけ.

S = input()

if S[0] == S[1] and S[1] == S[2]:
    print('Won')
else:
    print('Lost')

ABC189B - Alcoholic

3分で突破. double でやるのは嫌な予感がしたので、思考停止で Decimal で. 30秒考えれば100倍すればいいじゃんと気づいただろうに(笑).

from decimal import Decimal

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

c = 0
for i in range(N):
    V, P = map(Decimal, input().split())
    c += V * (P / 100)
    if c > X:
        print(i + 1)
        break
else:
    print(-1)

みんなXを100倍してたので、天の邪鬼で割り算で(笑).

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

c = 0
for i in range(N):
    V, P = map(int, input().split())
    c += V * P
    if (c + 99) // 100 > X:
        print(i + 1)
        break
else:
    print(-1)

ABC189D - Logical Expression

20分くらいで突破. C問題に一旦敗退してこちらを先に. 普通に AND のほうが OR より優先順位が高いんだろうと思いつつ問題文を読んだら優先順位が同じで、何この簡単な問題ってなった.

N, *S = open(0).read().split()

t = 1
f = 1
for s in S:
    if s == 'AND':
        f = t + f * 2
    elif s == 'OR':
        t = t * 2 + f
print(t)

公式解説も 2N が出てきているが、他の人の回答を見ても 2N が出てきている人が多いなあ.

N = int(input())
S = [input() for _ in range(N)]


def dfs(n):
    if n == 0:
        return 1

    if S[n - 1] == 'AND':
        return dfs(n - 1)
    elif S[n - 1] == 'OR':
        return 2 ** n + dfs(n - 1)


print(dfs(N))

ABC189C - Mandarin Orange

20分くらいで突破、WA1. N≤104 だから O(N2) だと駄目だよなと思いつつ、順位表を見ると正当者数が多いので、多分テストケースが制限いっぱいに攻めてないんだろうと投げたら AC が出て一安心. 公式解法も O(N2) で、そもそも制限時間も2秒ではなく、1.5秒だったことに気づいて目が白黒してる.

N, *A = map(int, open(0).read().split())

def f(x):
    result = 0
    i = 0
    j = 0
    while i < N:
        while j < N and x <= A[j]:
            j += 1
        result = max(result, x * (j - i))
        i = j + 1
        j = i
    return result

result = 0
for x in set(A):
    result = max(result, f(x))
print(result)

リストの中の最小要素で分割するを繰り返すと非常に高速に解けた. 上の解答は Python では TLE するので PyPy が必要だったが、こちらは Python でも AC する. とはいっても 1 から 104 まで順に並べた入力の処理に 3.5s かかるので、C問題のテストケースが弱いだけではある.

from sys import setrecursionlimit

setrecursionlimit(10 ** 6)

N, *A = map(int, open(0).read().split())


def f(a):
    n = len(a)
    m = min(a)
    result = m * n
    i = 0
    for j in range(n):
        if a[j] != m:
            continue
        if i < j:
            result = max(result, f(a[i:j]))
        i = j + 1
    if i < n:
        result = max(result, f(a[i:n]))
    return result


print(f(A))

上記の最悪 O(N2) で O(NlogN) なアルゴリズムを、セグ木を使って最悪 O(NlogN) で O(N) なアルゴリズムに改善してみた. 上記では最小の値とそのインデックスの取得が O(N) なところをセグ木を使って O(logN) に改善した.

from sys import setrecursionlimit


class SegmentTree:
    def __init__(self, size, op, e):
        self._op = op
        self._e = e
        self._size = size
        t = 1
        while t < size:
            t *= 2
        self._offset = t - 1
        self._data = [e] * (t * 2 - 1)

    def __getitem__(self, key):
        return self._data[self._offset + key]

    def __setitem__(self, key, value):
        op = self._op
        data = self._data
        i = self._offset + key
        data[i] = value
        while i >= 1:
            i = (i - 1) // 2
            data[i] = op(data[i * 2 + 1], data[i * 2 + 2])

    def build(self, iterable):
        op = self._op
        data = self._data
        data[self._offset:self._offset + self._size] = iterable
        for i in range(self._offset - 1, -1, -1):
            data[i] = op(data[i * 2 + 1], data[i * 2 + 2])

    def query(self, start, stop):
        def iter_segments(data, l, r):
            while l < r:
                if l & 1 == 0:
                    yield data[l]
                if r & 1 == 0:
                    yield data[r - 1]
                l = l // 2
                r = (r - 1) // 2
        op = self._op
        it = iter_segments(self._data, start + self._offset,
                           stop + self._offset)
        result = self._e
        for v in it:
            result = op(result, v)
        return result


setrecursionlimit(10 ** 6)

N, *A = map(int, open(0).read().split())

st = SegmentTree(N, min, (10 ** 18, -1))
st.build((A[i], i) for i in range(N))


def f(l, r):
    if l == r:
        return 0
    m, i = st.query(l, r)
    result = m * (r - l)
    result = min(result, f(l, i))
    result = min(result, f(i + 1, r))
    return result


print(f(0, N))

蟻本298ページに答えが載っていたが、イマイチ何をしているのかわからない(特にスタックの部分). ただまあ、どこまで左右に伸ばせるかという情報を再利用して高速化しているのは分かったので、高速な実装が組めた.

N, *A = map(int, open(0).read().split())

r = [0] * N
for i in range(N - 1, -1, -1):
    j = i + 1
    while j < N:
        if A[j] < A[i]:
            break
        j = r[j]
    r[i] = j

result = 0
l = [0] * N
for i in range(N):
    j = i
    while j > 0:
        if A[j - 1] < A[i]:
            break
        j = l[j - 1]
    l[i] = j
    result = max(result, A[i] * (r[i] - l[i]))
print(result)

最大値から順に伸ばしてよければ左右を結合していく方法でも解けた.

from sys import setrecursionlimit
from operator import itemgetter


def find(parent, i):
    t = parent[i]
    if t < 0:
        return i
    t = find(parent, t)
    parent[i] = t
    return t


def unite(parent, i, j):
    i = find(parent, i)
    j = find(parent, j)
    if i == j:
        return
    parent[j] += parent[i]
    parent[i] = j


setrecursionlimit(10 ** 6)

N, *A = map(int, open(0).read().split())

result = 0
parent = [-1] * (N + 1)
for i, _ in sorted(enumerate(A), reverse=True, key=itemgetter(1)):
    for j in [i - 1, i + 1]:
        if j < 0 or j >= N:
            continue
        if A[j] < A[i]:
            continue
        unite(parent, i, j)
    result = max(result, A[i] * -parent[find(parent, i)])
print(result)

ABC189E - Rotate and Flip

56分で突破. アフィン変換、なんでしたっけ?(まて)

情報学科卒にも関わらずアフィン変換という単語は出てこなかったものの、行列とは気づかずに行列の遷移を書けたおかげで AC できた.

from sys import stdin

readline = stdin.readline

N = int(readline())
XY = [tuple(map(int, readline().split())) for _ in range(N)]
M = int(readline())
op = [readline() for _ in range(M)]
Q = int(readline())
AB = [tuple(map(int, readline().split())) for _ in range(Q)]

# (x, y)
# 1 -> (y, -x)
# 2 -> (-y, x)
# 3 p -> (2p - x, y)
# 4 p -> (x, 2p - y)


def e(x, i):
    result = x[0]
    if x[1] == 1:
        result += XY[i][x[2]]
    elif x[1] == -1:
        result -= XY[i][x[2]]
    return result


x = (0, 1, 0)
y = (0, 1, 1)
applied = 0
result = [None] * Q
for a, b, i in sorted(((AB[i][0], AB[i][1], i) for i in range(Q)), key=lambda x: x[0]):
    while applied < a:
        o = op[applied]
        if o[0] == '1':
            t = x
            x = (y[0], y[1], y[2])
            y = (-t[0], -t[1], t[2])
        if o[0] == '2':
            t = x
            x = (-y[0], -y[1], y[2])
            y = (t[0], t[1], t[2])
        if o[0] == '3':
            p = int(o[2:])
            x = (2 * p - x[0], -x[1], x[2])
        if o[0] == '4':
            p = int(o[2:])
            y = (2 * p - y[0], -y[1], y[2])
        applied += 1
    result[i] = '%d %d' % (e(x, b - 1), e(y, (b - 1)))
print(*result, sep='\n')
1
1
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
1
1