珂朵莉树

珂朵莉树

简要

珂朵莉树的核心操作:split

实际很简单,一个集合中,有一部分需要修改,而另一部分不需要修改,就把集合拆开,拆成两部分。

珂朵莉树的推平操作:assign

珂朵莉树的复杂度是由assign保证的。

由于数据随机,有14\frac{1}{4}41的操作为assign。

set的大小快速下降,最终趋于log⁡n\log nlogn ,使得这种看似暴力无比的数据结构复杂度接近mlog⁡nm\log nmlogn

struct  node {int l, r;mutable ll v;node(int _l, int _r = -1, ll _v = 0) : l(_l), r(_r), v(_v) {}bool operator < (const node &t) const {return l < t.l;}
};set<node>::iterator split(int pos) {set<node>::iterator it = st.lower_bound(node(pos));if (it != st.end() && it->l == pos) {return it;}it--;int l = it->l, r = it->r;ll v = it->v;st.erase(it);st.insert(node(l, pos - 1, v));return st.insert(node(pos, r, v)).first;
}void assign(int l, int r, ll value) {set<node>::iterator itr = split(r + 1), itl = split(l);st.erase(itl, itr);st.insert(node(l, r, value));
}

下面是一些可用珂朵莉树acacac题目,好多题目都卡珂朵莉树了,所以只能迫不得已写毒瘤线段树。

CF896C Willem, Chtholly and Seniorious

#include <bits/stdc++.h>
#define int long longusing namespace std;typedef long long ll;
typedef pair<int, int> pii;struct  node {int l, r;mutable ll v;node(int _l, int _r = -1, ll _v = 0) : l(_l), r(_r), v(_v) {}bool operator < (const node &t) const {return l < t.l;}
};set<node> st;set<node>::iterator split(int pos) {set<node>::iterator it = st.lower_bound(node(pos));if (it != st.end() && it->l == pos) {return it;}it--;int l = it->l, r = it->r;ll v = it->v;st.erase(it);st.insert(node(l, pos - 1, v));return st.insert(node(pos, r, v)).first;
}void add(int l, int r, ll value) {set<node>::iterator itr = split(r + 1), itl = split(l);for (; itl != itr; itl++) {itl->v += value;}
}void assign(int l, int r, ll value) {set<node>::iterator itr = split(r + 1), itl = split(l);st.erase(itl, itr);st.insert(node(l, r, value));
}ll rank_k(int l, int r, int k, bool reversed = false) {if (reversed) {k = r - l + 2 - k;}set<node>::iterator itr = split(r + 1), itl = split(l);vector<pii> ans;for (; itl != itr; itl++) {ans.push_back(make_pair(itl->v, itl->r - itl->l + 1));}sort(ans.begin(), ans.end());for (auto it : ans) {k -= it.second;if (k <= 0) {return it.first;}}return -1;
}ll quick_pow(ll a, int n, int mod) {ll ans = 1;a %= mod;while (n) {if (n & 1) {ans = ans * a % mod;}a = a * a % mod;n >>= 1;}return ans;
}ll sum(int l, int r, int n, int mod) {set<node>::iterator itr = split(r + 1), itl = split(l);ll ans = 0;for (; itl != itr; itl++) {ans = (ans + 1ll * (itl->r - itl->l + 1) * quick_pow(itl->v, n, mod) % mod) % mod;}return ans;
}int n, m, seed, vmax;const int mod = 1e9 + 7;int rnd() {int ret = seed;seed = (1ll * seed * 7 + 13) % mod;return ret;
}signed main() {// freopen("in.txt", "r", stdin);// freopen("out.txt", "w", stdout);// ios::sync_with_stdio(false), cin.tie(0), cout.tie(0);scanf("%lld %lld %lld %lld", &n, &m, &seed, &vmax);for (int i = 1; i <= n; i++) {st.insert(node(i, i, rnd() % vmax + 1));}for (int i = 1; i <= m; i++) {int op = rnd() % 4 + 1, l = rnd() % n + 1, r = rnd() % n + 1, x, y;if (l > r) {swap(l, r);}if (op == 3) {x = rnd() % (r - l + 1) + 1;}else {x = rnd() % vmax + 1;}if (op == 4) {y = rnd() % vmax + 1;}if (op == 1) {add(l, r, x);}else if (op == 2) {assign(l, r, x);}else if (op == 3) {printf("%lld\n", rank_k(l, r, x));}else {printf("%lld\n", sum(l, r, x, y));}}return 0;
}

CF915E Physical Education Lessons

#include <bits/stdc++.h>using namespace std;struct  node {int l, r;mutable int v;node(int _l, int _r = -1, int _v = 0) : l(_l), r(_r), v(_v) {}bool operator < (const node &t) const {return l < t.l;}
};int n, m;set<node> st;set<node>::iterator split(int pos) {set<node>::iterator it = st.lower_bound(node(pos));if (it != st.end() && it->l == pos) {return it;}it--;int l = it->l, r = it->r, v = it->v;st.erase(it);st.insert(node(l, pos - 1, v));return st.insert(node(pos, r, v)).first;
}void assign(int l, int r, int value) {set<node>::iterator itr = split(r + 1), itl = split(l), it = itl;for (; itl != itr; itl++) {n -= (itl->r - itl->l + 1) * itl->v;}st.erase(it, itr);st.insert(node(l, r, value));n += (r - l + 1) * value;
}int main() {// freopen("in.txt", "r", stdin);// freopen("out.txt", "w", stdout);// ios::sync_with_stdio(false), cin.tie(0), cout.tie(0);scanf("%d %d", &n, &m);st.insert(node(1, n, 1));for (int i = 1; i <= m; i++) {int l, r, v;scanf("%d %d %d", &l, &r, &v);assign(l, r, v & 1 ^ 1);printf("%d\n", n);}return 0;
}

CF343D Water Tree

#include <bits/stdc++.h>using namespace std;const int N = 5e5 + 10;int head[N], to[N << 1], nex[N << 1], cnt = 1;int n, m, tot, son[N], sz[N], fa[N], top[N], rk[N], id[N], dep[N];struct node {int l, r;mutable int v;node(int _l, int _r = -1, int _v = 0) : l(_l), r(_r), v(_v) {}bool operator < (const node &t) const {return l < t.l;}
};void add(int x, int y) {to[cnt] = y;nex[cnt] = head[x];head[x] = cnt++;
}void dfs1(int rt, int f) {sz[rt] = 1, fa[rt] = f, dep[rt] = dep[f] + 1;for (int i = head[rt]; i; i = nex[i]) {if (to[i] == f) {continue;}dfs1(to[i], rt);sz[rt] += sz[to[i]];if (!son[rt] || sz[to[i]] > sz[son[rt]]) {son[rt] = to[i];}}
}void dfs2(int rt, int tp) {top[rt] = tp, rk[++tot] = rt, id[rt] = tot;if (!son[rt]) {return ;}dfs2(son[rt], tp);for (int i = head[rt]; i; i = nex[i]) {if (to[i] == fa[rt] || to[i] == son[rt]) {continue;}dfs2(to[i], to[i]);}
}set<node> st;auto split(int pos) {auto it = st.lower_bound(node(pos));if (it != st.end() && it->l == pos) {return it;}it--;int l = it->l, r = it->r, v = it->v;st.erase(it);st.insert(node(l, pos - 1, v));return st.insert(node(pos, r, v)).first;
}void assign(int l, int r, int value) {auto itr = split(r + 1), itl = split(l);st.erase(itl, itr);st.insert(node(l, r, value));
}int main() {// freopen("in.txt", "r", stdin);// freopen("out.txt", "w", stdout);// ios::sync_with_stdio(false), cin.tie(0), cout.tie(0);scanf("%d", &n);for (int i = 1; i < n; i++) {int x, y;scanf("%d %d", &x, &y);add(x, y);add(y, x);}dfs1(1, 0);dfs2(1, 1);st.insert(node(1, n, 0));scanf("%d", &m);for (int i = 1; i <= m; i++) {int op, u;scanf("%d %d", &op, &u);if (op == 1) {assign(id[u], id[u] + sz[u] - 1, 1);}else if (op == 2) {int x = 1, y = u;while (top[x] != top[y]) {if (dep[top[x]] < dep[top[y]]) {swap(x, y);}assign(id[top[x]], id[x], 0);x = fa[top[x]];}if (dep[x] > dep[y]) {swap(x, y);}assign(id[x], id[y], 0);}else {auto it = st.lower_bound(node(id[u]));if (it->l > id[u]) {it--;}printf("%d\n", it->v);}}return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/314023.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

2019-03-13-算法-进化(鸡蛋掉落)

题目描述 你将获得 K 个鸡蛋&#xff0c;并可以使用一栋从 1 到 N 共有 N 层楼的建筑。 每个蛋的功能都是一样的&#xff0c;如果一个蛋碎了&#xff0c;你就不能再把它掉下去。 你知道存在楼层 F &#xff0c;满足 0 < F < N 任何从高于 F 的楼层落下的鸡蛋都会碎&…

P3810 【模板】三维偏序(陌上花开)

P3810 【模板】三维偏序&#xff08;陌上花开&#xff09; cdq分治的模板题目&#xff0c;用cdq分治处理点对的问题&#xff0c;首先按照第一维排序&#xff0c;然后分治&#xff0c;这样就可以每次考虑二三维即可&#xff0c;然后就可以利用树状数组求解了&#xff0c;然后一…

用pythonnet为计算机视觉做图像整理

中国的.NETer是国内技术的另类&#xff0c;当他们强调.NET也可以做啥啥时都会给别的技术藐视&#xff0c;毕竟主流都不用.NET。本人这几年其实花在.NET时间也少&#xff0c;都投入在Python/Go社区。可我还是有点工作外的寄托&#xff0c;就是让.NET也有一个很好的推广&#xff…

ABP虚拟文件系统(VirtualFileSystem)实例------定制菜单栏显示用户姓名

ABP默认的MVC启动模板在登录后, 右上角显示的是用户名:如果想让它显示用户的姓名该如何做呢?这就需要用到ABP一个非常强大的功能------虚拟文件系统.前期准备使用ABP CLI创建一个名为AbpStudy的ASP.NET MVC项目:abp new AbpStudy关于MVC的启动模板可以看文档, 这里就不赘述.虚…

序列计数(动态规划/自动机/前缀和优化)

序列计数 对于一个小写字母的序列每次可以将相邻两个不同的小写字母都变为二者之一&#xff0c;可以进行无限次这样的操作&#xff0c;求解可以产生多少种不同的序列。 首先我们不能考虑操作&#xff0c;而是考虑合法序列&#xff0c;显然最后会形成若干个区间&#xff0c;然…

概率期望dp

概率期望 LOOPS dp[i][j]dp[i][j]dp[i][j]表示从i,ji, ji,j到r,cr, cr,c的期望&#xff0c;有dp[i][j]p0dp[i][j]p1dp[i][j1]p2dp[i1][j]2dp[i][j] p_0 \times dp[i][j] p_1 \times dp[i][j 1] p_2 \times dp[i 1][j] 2dp[i][j]p0​dp[i][j]p1​dp[i][j1]p2​dp[i1][j]2…

2019-03-13-算法-进化(验证回文串)

题目描述 给定一个字符串&#xff0c;验证它是否是回文串&#xff0c;只考虑字母和数字字符&#xff0c;可以忽略字母的大小写。 说明&#xff1a;本题中&#xff0c;我们将空字符串定义为有效的回文串。 示例 1: 输入: "A man, a plan, a canal: Panama" 输出: t…

.NET中国峰会 参与意愿调查

2014年微软组织成立.NET基金会&#xff0c;微软在成为主要的开源参与者的道路上又前进了一步。2014年以来已经有众多知名公司加入.NET基金会, 仅在平台项目中&#xff0c;.NET平台上有87&#xff05;贡献者其实不在Microsoft工作。将.NET基金会变成一个更加多样化和成员驱动的组…

2019-03-14-算法-进化(两个数组的交集 II)

题目描述 给定两个数组&#xff0c;编写一个函数来计算它们的交集。 示例 1: 输入: nums1 [1,2,2,1], nums2 [2,2] 输出: [2,2]示例 2: 输入: nums1 [4,9,5], nums2 [9,4,9,8,4] 输出: [4,9]说明&#xff1a; 输出结果中每个元素出现的次数&#xff0c;应与元素在两个数…

方案计数(带修计数题/线段树)

方案计数 对于一个n个队员&#xff0c;每个队员有一个权值Vi&#xff0c;然后每次选择三个权值相同的A类队员&#xff0c;两个权值小于A类的B类队员&#xff0c;并且B类队员要在A类队员两侧。 Q次操作&#xff0c;每次限制或解除限制一个队员成为A类队员。 首先枚举中间的A类…

线性基专题

线性基 P3812 【模板】线性基 #include <bits/stdc.h>using namespace std;typedef long long ll;struct linearbasis {ll base[64], flag, cnt;void add(ll x) {for(int i 62; ~i; i--) {if(x >> i & 1) {if(!base[i]) {base[i] x;return ;}x ^ base[i];}…

听我的!美国科技公司这样做Code Review

Code Review&#xff0c;在当代的软件开发中占有重要的一环。虽然国内各大主流公司都已经参照国外同行设立了比较严格的Code Review机制&#xff0c;但是还是有好多大型软件公司以及中小型软件公司还未推行这一重要制度。那么在美国的科技企业中&#xff0c;Code Review推行的怎…

无向图三元环计数

无向图三元环计数 这个做法的思想还是很巧妙的&#xff0c;首先我们考虑枚举&#xff0c;暴力的方法就是枚举三个点O(n3)O(n^3)O(n3)&#xff0c;枚举一个点然后枚举出边&#xff0c;然后再枚举出点的出边&#xff0c;然后考虑这个做法的复杂度。对于每条边分析&#xff0c;它…

2019-03-14-算法-进化(移动零)

题目描述 给定一个数组 nums&#xff0c;编写一个函数将所有 0 移动到数组的末尾&#xff0c;同时保持非零元素的相对顺序。 示例: 输入: [0,1,0,3,12] 输出: [1,3,12,0,0]说明: 必须在原数组上操作&#xff0c;不能拷贝额外的数组尽量减少操作次数 /*** 思路1&#xff1a;…

美味果冻(牛客练习赛53B)

美味果冻 ∑i1n∑j1ii⌊ij⌋j∑i1n∑jinj⌊ji⌋i\sum_{i 1} ^{n} \sum_{j 1} ^{i} i \times \lfloor \frac{i}{j} \rfloor ^ j\\ \sum_{i 1} ^{n} \sum_{j i} ^{n} j \times \lfloor \frac{j}{i} \rfloor ^ i\\ i1∑n​j1∑i​i⌊ji​⌋ji1∑n​ji∑n​j⌊ij​⌋i 接下来只…

程序员过关斩将--互联网人必备知识cookie和session认证

菜菜&#xff0c;上次你说的cookie和session的文章&#xff0c;我觉得不太具体那你想怎么样具体呢&#xff1f;我自己从网上查了一下&#xff0c;很多关于cookie和session认证的&#xff0c;能不能给我讲讲用户认证呀&#xff0c;可以呀这样我下次再去面试&#xff0c;有可能会…

2019-03-15-算法-进化(两数之和)

题目描述 给定一个整数数组 nums 和一个目标值 target&#xff0c;请你在该数组中找出和为目标值的那 两个 整数&#xff0c;并返回他们的数组下标。 你可以假设每种输入只会对应一个答案。但是&#xff0c;你不能重复利用这个数组中同样的元素。 示例: 给定 nums [2, 7, 1…

51nod 1847 奇怪的数学题(数论/min25筛/杜教筛/斯特林数)

51nod 1847 奇怪的数学题 求解∑i1n∑j1nsgcd(i,j),sgcd\sum_{i1}^n\sum_{j1}^nsgcd(i,j),sgcd∑i1n​∑j1n​sgcd(i,j),sgcd表示次大公约数,n≤1010n\le{10^{10}}n≤1010 那么首先有sgcd(i,j)gcd(i,j)/mn(gcd(i,j))sgcd(i,j)gcd(i,j)/mn(gcd(i,j))sgcd(i,j)gcd(i,j)/mn(gcd(…

.NET Core ASP.NET Core Basic 1-2 控制反转与依赖注入

本节内容为控制反转与依赖注入简介控制反转IOC这个内容事实上在我们的C#高级篇就已经有所讲解&#xff0c;控制反转是一种设计模式&#xff0c;你可以这样理解控制反转&#xff0c;假设有一个人他有一部A品牌手机&#xff0c;他用手机进行听歌、打游戏&#xff0c;那么你可以创…

2019-03-15-算法-进化(有效的数独)

题目描述 判断一个 9x9 的数独是否有效。只需要根据以下规则&#xff0c;验证已经填入的数字是否有效即可。数字 1-9 在每一行只能出现一次。 数字 1-9 在每一列只能出现一次。 数字 1-9 在每一个以粗实线分隔的 3x3 宫内只能出现一次。上图是一个部分填充的有效的数独。 数独…