モノイド $(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) var seg = initSegTree[S](n:int, op:(S,S)->S, e:()->S):segtree[S]
var seg = SegTreeType(S, op:(S,S)->S, e:()->S).init(n)
var seg = SegTree.getType(S, op:(S,S)->S, e:()->S).init(n)
(2) var seg = initSegTree[S](v:seq[S], op:(S,S)->S, e:()->S):segtree[S]
var seg = SegTreeType(S, op:(S,S)->S, e:()->S).init(v)
var seg = SegTree.getType(S, op:(S,S)->S, e:()->S).init(v)
S
op(a:S, b:S):S
e():S
を定義する必要があります。例として、Range Min Queryならば
proc op(a:int, b:int):int =
return min(a, b)
proc e():int =
return 10^9
var seg = initSegTree[int](10, seg, e)
のようになります。
n
の数列 a
を作ります。初期値は全部e()
です。n = v.len
の数列 a
を作ります。v
の内容が初期値となります。詳しくは、使用例や こちら も参照してください。
制約
計算量
type st_type = SegTreeType[S](op:(S,S)->S, e:()->S)
コンストラクタは呼ばずにタイプだけ設定できます。st_type.init(v:seq[int])
といった具合にタイプからコンストラクタを呼び出したり、var v:seq[st_type]
といった具合にsegtreeの配列を宣言するのにお使いください。
(背景)Nimはいろいろと癖のある言語でsegtreeのobject構造体はマニアック(笑)な構造をしているので直にタイプを呼ぶのは推奨されません。上記のSegTreeTypeのテンプレートをお使いください。
seg.set(p:int, x:S):void
seg[p:int] = x:S
a[p]
に x
を代入します。
制約
計算量
seg.get(p:int):S
seg[p:int]:S
a[p]
を返します。
制約
計算量
seg.prod(l..<r):S
seg[l..<r]:S
op(a[l], ..., a[r - 1])
を、モノイドの性質を満たしていると仮定して計算します。$l = r$ のときは e()
を返します。
制約
計算量
seg.all_prod():S
op(a[0], ..., a[n - 1])
を計算します。$n = 0$ のときは e()
を返します。
計算量
(1) seg.max_right<f>(l:int):int
(2💻) seg.max_right[F](l:int, f:F):int
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) seg.min_left[f](r:int):int
(2💻) seg.min_left[F](r:int, f:F):int
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], ..., a[r - 1])) = false
f
が単調だとすれば、f(op(a[l], a[l + 1], ..., a[r - 1])) = true
となる最小の l
、と解釈することが可能です。
制約
f
を同じ引数で呼んだ時、返り値は等しい(=副作用はない)f(e()) = true
計算量