线段树合并模板
1.并查集模板2.求质因数模板3.二分图最大匹配模板(匈牙利算法)4.欧拉函数模板5.ST表模板6.快速幂模板7.字典树模板8.矩阵模板9.Dijkstra单源最短路模板10.最近公共祖先模板(LCA)11.拓扑排序模板12.区间素数筛模板13.Kruskal和Prim模板14.树状数组模板15.二维坐标离散化模板16.单点修改区间查最值-树状数组模板17.KMP模板18.二叉搜索树模板19.DIjkstra进阶模板 路径记录 按权重(结点数最小等)记录20.判断负环模板21.Exgcd 模板22.压位高精度模板23.线段树模板24.扫描线模板25.莫队模板26.带修莫队模板27.SCC缩点模板28.取模+组合数29.FFT 高精度乘法模板30.字符串自然溢出哈希/单哈希/双哈希模板31.树模板32.dsu on tree 模板33.线段树模板重制34.主席树模板35.大数质因数分解模板
36.线段树合并模板
37.int128输入输出流38.Meissel_Lehmer模板39.浮点高精度40.自适应辛普森法41.unordered_map随机底数种子template<class Node> struct PersidentSegmentTree { #define lc(u) tr[u].l #define rc(u) tr[u].r const int n; int tot = 0; vector<Node> tr; vector<int> root; PersidentSegmentTree(): n(0) {} PersidentSegmentTree(int n_): n(n_) { int N = (n << 6) + 10; tr.reserve(N); root.reserve(N); tr.resize(N); root.resize(N); } PersidentSegmentTree(vector<int>& a): PersidentSegmentTree(a.size() - 1) { function<void(int&, int, int)> build = [&](int& now, int l, int r) { now = ++ tot; if (l == r) { return ; } int m = (l + r) >> 1; build(lc(now), l, m); build(rc(now), m + 1, r); }; build(root[0], 1, n); } //上传 void pushup(int u) { if (tr[lc(u)].Sum >= tr[rc(u)].Sum) { tr[u].Sum = tr[lc(u)].Sum; tr[u].pos = tr[lc(u)].pos; } else { tr[u].Sum = tr[rc(u)].Sum; tr[u].pos = tr[rc(u)].pos; } } //动态开点/更新树结点 void insert(int& now, int l, int r, int pos, int w) { if (!now) now = ++ tot; if (l == r) { tr[now].Sum += w; tr[now].pos = pos; return; } int m = l + r >> 1; if (pos <= m) insert(lc(now), l, m, pos, w); else insert(rc(now), m + 1, r, pos, w); pushup(now); } //开新前缀树,last代表前缀,now代表新树根 void insert(int& now, int last, int l, int r, int pos, int w) { now = ++ tot; tr[now] = tr[last]; if (l == r) { return; } int m = l + r >> 1; if (pos <= m) insert(lc(now), lc(last), l, m, pos, w); else insert(rc(now), rc(last), m + 1, r, pos, w); } //单点 int query(int now, int l, int r, int pos) { if (l == r) { return tr[now].Sum; } int m = l + r >> 1; if (pos <= m) return query(lc(now), l, m, pos); else return query(rc(now), m + 1, r, pos); } //区间查询 [u,v]->[root[l-1],root[r]] int query(int u, int v, int l, int r, int pos) { if (l == r) { return tr[u].Sum; } int m = l + r >> 1; if (pos <= m) return query(lc(u), lc(v), l, m, pos); else return query(rc(u), rc(v), m + 1, r, pos); } void merge(int &u, int v, int l, int r) { if (!u || !v) { u = u + v; return; } if (l == r) { tr[u].Sum += tr[v].Sum; return ; } int m = l + r >> 1; merge(lc(u), lc(v), l, m); merge(rc(u), rc(v), m + 1, r); pushup(u); } }; struct Node { int l, r; int Sum = 0, pos = 0; };
本文作者:Ke_scholar
本文链接:https://www.cnblogs.com/Kescholar/p/18350415
版权声明:本作品采用知识共享署名-非商业性使用-禁止演绎 2.5 中国大陆许可协议进行许可。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步