🚧 This website is under construction. 🚧

ABC174 F - Range Set Query

https://atcoder.jp/contests/abc174/tasks/abc174_f
水色上位。データ構造。

クエリに対してオフラインに回答することを考える。 まずはスケジューリング問題と同様にまずはクエリを右端の値でソートしておき、この順番に回答する。

左端のボールから順番に見ていくが、その時同じ色のボールは一番右側に位置するもののみ管理する。(区間の右端でソートしており、区間の左端のインデックスより大きいものにのみ興味があるため)。各色のボールにインデックスに対応する部分にフラグを立てておけば、BIT により区間和を計算することによって各クエリに回答できる。

import sys
 
input = sys.stdin.readline
 
 
def add(p, x):
    p += 1
    while p <= n:
        bit[p - 1] += x
        p += p & -p
 
 
def sum(left, right):
    return _sum(right) - _sum(left)
 
 
def _sum(r):
    s = 0
    while r > 0:
        s += bit[r - 1]
        r -= r & -r
 
    return s
 
 
n, q = map(int, input().split())
c = list(map(lambda x: int(x) - 1, input().split()))
lr = [list(map(lambda x: int(x) - 1, input().split())) for _ in range(q)]
 
res = [None] * q
 
# balls[i] := 最も右端に位置する色iのボールのインデックス
balls = [None] * n
 
# bit[i] := インデックスiに条件を満たすボールが存在する
bit = [0] * n
 
pre_r = 0
for i in sorted(range(q), key=lambda i: lr[i][1]):
    l, r = lr[i]
    for j in range(pre_r, r + 1):
        if not balls[c[j]] is None:
            add(balls[c[j]], -1)
        balls[c[j]] = j
        add(balls[c[j]], 1)
    res[i] = sum(l, r + 1)
    pre_r = r
 
print(*res)
 

定数倍との勝負であり、何回か提出する必要がある。