This documentation is automatically generated by online-judge-tools/verification-helper
#include "Data_structure/Segtree.hpp"
セグメント木 モノイドを渡す
集合 $S$ とその二項演算 $\cdot : S \times S \to S$ で、
を満たすとき、$(S, \cdot, e)$ をモノイドという。
例えば加法モノイドなら、次のように書く。
struct Add_M {
using T = long long;
static T e() { return 0; }
static T op(T x, T y) { return x + y; }
};
T
はモノイドの型名。e()
は単位元、op(T x, T y)
は二項演算を返すようにする。
(1) Segtree<Monoid> seg(int N)
(2) Segtree<Monoid> seg(vector<Monoid::T> v)
N
ですべてMonoid::e()
の数列で初期化v
の内容で初期化void seg.set(int p, Monoid::T x)
a[p]
にx
を代入する。
Monoid::T seg.get(int p)
a[p]
の値を取得する。
Monoid::T seg.prod(int l, int r)
Monoid::op(a[l], ..., a[r-1])
を返す。$l = r$の時はMonoid::e()
が返る。
Monoid::T seg.all_prod()
Monoid::op(a[0], ..., a[n-1])
を返す。
TODO: セグ木上の二分探索
struct Add_M {
using T = long long;
static T e() { return 0; }
static T op(T x, T y) { return x + y; }
};
struct Max_M {
using T = long long;
static T e() { return -INF; }
static T op(T x, T y) { return max(x, y); }
};
struct Min_M {
using T = long long;
static T e() { return INF; }
static T op(T x, T y) { return min(x, y); }
};
struct Monoid {
using T = T;
static T e() { return e; }
static T op(T x, T y) { return op(x, y); }
}
template<class M> struct Segtree {
public:
using S = typename M::T;
Segtree() : Segtree(0) {}
Segtree(int n) : Segtree(vector<S> (n, M::e())) {}
Segtree(const vector<S> &v) : n(int(v.size())) {
while((1 << log) < n) log++;
size = 1 << log;
d = vector<S> (2 * size, M::e());
for(int i = 0; i < n; i++) d[size + i] = v[i];
for(int i = size - 1; i >= 1; i--) update(i);
}
void set(int p, S x) {
assert(0 <= p && p < n);
p += size;
d[p] = x;
for(int i = 1; i <= log; i++) update(p >> i);
}
S get(int p) {
assert(0 <= p && p < n);
return d[p + size];
}
S prod(int l, int r) {
assert(0 <= l && l <= r && r <= n);
S sml = M::e(), smr = M::e();
l += size; r += size;
while(l < r) {
if(l & 1) sml = M::op(sml, d[l++]);
if(r & 1) smr = M::op(d[--r], smr);
l >>= 1; r >>= 1;
}
return M::op(sml, smr);
}
S all_prod(){ return d[1]; }
private:
int n, size, log = 0;
vector<S> d;
void update(int k){ d[k] = M::op(d[k * 2], d[k * 2 + 1]); }
};
#line 1 "Data_structure/Segtree.hpp"
template<class M> struct Segtree {
public:
using S = typename M::T;
Segtree() : Segtree(0) {}
Segtree(int n) : Segtree(vector<S> (n, M::e())) {}
Segtree(const vector<S> &v) : n(int(v.size())) {
while((1 << log) < n) log++;
size = 1 << log;
d = vector<S> (2 * size, M::e());
for(int i = 0; i < n; i++) d[size + i] = v[i];
for(int i = size - 1; i >= 1; i--) update(i);
}
void set(int p, S x) {
assert(0 <= p && p < n);
p += size;
d[p] = x;
for(int i = 1; i <= log; i++) update(p >> i);
}
S get(int p) {
assert(0 <= p && p < n);
return d[p + size];
}
S prod(int l, int r) {
assert(0 <= l && l <= r && r <= n);
S sml = M::e(), smr = M::e();
l += size; r += size;
while(l < r) {
if(l & 1) sml = M::op(sml, d[l++]);
if(r & 1) smr = M::op(d[--r], smr);
l >>= 1; r >>= 1;
}
return M::op(sml, smr);
}
S all_prod(){ return d[1]; }
private:
int n, size, log = 0;
vector<S> d;
void update(int k){ d[k] = M::op(d[k * 2], d[k * 2 + 1]); }
};