モノイド $(S, \cdot: S \times S \to S, e \in S)$、つまり
を満たす代数構造に対し使用できるデータ構造です。
長さ $N$ の $S$ の配列に対し、
を $O(\log N)$ で行うことが出来ます。詳細な要件は Appendix を参照してください。
また、このライブラリはオラクルとしてop, e
の2種類を使用しますが、これらが定数時間で動くものと仮定したときの計算量を記述します。オラクル内部の計算量が $O(f(n))$ である場合はすべての計算量が $O(f(n))$ 倍となります。
(1) segtree<S, op, e> seg(int n)
(2) segtree<S, op, e> seg(vector<S> v)
S
S op(S a, S b)
S e()
を定義する必要があります。例として、Range Min Queryならば
int op(int a, int b) {
return min(a, b);
}
int e() {
return (int)(1e9);
}
segtree<int, op, e> seg(10);
のようになります。
n
の数列 a
を作ります。初期値は全部e()
です。n = v.size()
の数列 a
を作ります。v
の内容が初期値となります。詳しくは、使用例や こちら も参照してください。
制約
計算量
void seg.set(int p, S x)
a[p]
に x
を代入します。
制約
計算量
S seg.get(int p)
a[p]
を返します。
制約
計算量
S seg.prod(int l, int r)
op(a[l], ..., a[r - 1])
を、モノイドの性質を満たしていると仮定して計算します。$l = r$ のときは e()
を返します。
制約
計算量
S seg.all_prod()
op(a[0], ..., a[n - 1])
を計算します。$n = 0$ のときは e()
を返します。
計算量
(1) int seg.max_right<f>(int l)
(2💻) int seg.max_right<F>(int l, F f)
bool f(S x)
を定義する必要があります。segtreeの上で二分探索をします。 S
を引数にとりbool
を返す関数オブジェクトを渡して使用します。 以下の条件を両方満たす r
を(いずれか一つ)返します。
r = l
もしくは f(op(a[l], a[l + 1], ..., a[r - 1])) = true
r = n
もしくは f(op(a[l], a[l + 1], ..., a[r])) = false
f
が単調だとすれば、f(op(a[l], a[l + 1], ..., a[r - 1])) = true
となる最大の r
、と解釈することが可能です。
制約
f
を同じ引数で呼んだ時、返り値は等しい(=副作用はない)f(e()) = true
計算量
(1) int seg.min_left<f>(int r)
(2💻) int seg.min_left<F>(int r, F f)
bool f(S x)
を定義する必要があります。segtreeの上で二分探索をします。 S
を引数にとりbool
を返す関数オブジェクトを渡して使用します。 以下の条件を両方満たす l
を(いずれか一つ)返します。
l = r
もしくは f(op(a[l], a[l + 1], ..., a[r - 1])) = true
l = 0
もしくは f(op(a[l - 1], a[l + 1], ..., a[r - 1])) = false
f
が単調だとすれば、f(op(a[l], a[l + 1], ..., a[r - 1])) = true
となる最小の l
、と解釈することが可能です。
制約
f
を同じ引数で呼んだ時、返り値は等しい(=副作用はない)f(e()) = true
計算量