清明梦超能力者黄YY

这题有点像【雨天的尾巴】【永无乡】的结合版本,树上差分,线段树合并,权值线段树查找第kkk大。

对于操作iii,我们可以对u−>vu->vu−>v路径上的点,iii的权值加上111,然后线段树合并,查找第kkk大就好了。

#include <bits/stdc++.h>using namespace std;const int N = 1e5 + 10;int head[N], to[N << 1], nex[N << 1], cnt = 1;int n, m, k, ans[N], root[N], value[N];int fa[N], top[N], dep[N], son[N], sz[N];int ls[N * 60], rs[N * 60], sum[N * 60], tot;void add(int x, int y) {to[cnt] = y;nex[cnt] = head[x];head[x] = cnt++;
}void dfs1(int rt, int f) {fa[rt] = f, dep[rt] = dep[f] + 1, sz[rt] = 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[son[rt]] < sz[to[i]]) {son[rt] = to[i];}}
}void dfs2(int rt, int tp) {top[rt] = tp;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]);}
}int lca(int x, int y) {while (top[x] != top[y]) {if (dep[top[x]] < dep[top[y]]) {swap(x, y);}x = fa[top[x]];}return dep[x] < dep[y] ? x : y;
}void push_up(int rt) {sum[rt] = sum[ls[rt]] + sum[rs[rt]];
}void update(int &rt, int l, int r, int x, int value) {if (!rt) {rt = ++tot;}if (l == r) {sum[rt] += value;return ;}int mid = (l + r) >> 1;if (x <= mid) {update(ls[rt], l, mid, x, value);}if (x > mid) {update(rs[rt], mid + 1, r, x, value);}push_up(rt);
}int merge(int x, int y, int l, int r) {if (x == 0 || y == 0) {return x | y;}if (l == r) {sum[x] += sum[y];return x;}int mid = (l + r) >> 1;ls[x] = merge(ls[x], ls[y], l, mid);rs[x] = merge(rs[x], rs[y], mid + 1, r);push_up(x);return x;
}int find_k_th(int rt, int l, int r, int k) {if (l == r) {return l;}int mid = (l + r) >> 1;if (k > sum[ls[rt]]) {return find_k_th(rs[rt], mid + 1, r, k - sum[ls[rt]]);}return find_k_th(ls[rt], l, mid, k);
}void dfs(int rt, int fa) {for (int i = head[rt]; i; i = nex[i]) {if (to[i] == fa) {continue;}dfs(to[i], rt);root[rt] = merge(root[rt], root[to[i]], 1, n);}if (k <= sum[root[rt]]) {ans[rt] = find_k_th(root[rt], 1, n, sum[root[rt]] - k + 1);}
}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 %d", &n, &m, &k);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);for (int i = 1; i <= m; i++) {int u, v, c;scanf("%d %d %d", &u, &v, &value[i]);int f = lca(u, v), ff = fa[f];update(root[u], 1, n, i, 1);update(root[v], 1, n, i, 1);update(root[f], 1, n, i, -1);if (ff) {update(root[ff], 1, n, i, -1);}}dfs(1, 0);for (int i = 1; i <= n; i++) {printf("%d%c", value[ans[i]], i == n ? '\n' : ' ');}return 0;
}

异或树

异或操作,容易想到拆位,然后依次算贡献,然后注意同一颗子树上相同的值如果出现了偶数次要相消,线段树合并一下就好了。

#include <bits/stdc++.h>using namespace std;typedef pair<int, int> pii;
typedef long long ll;const int N = 1e5 + 10;int head[N], to[N << 1], nex[N << 1], cnt = 1;int n, m, value[N], root[N];int ls[N * 33], rs[N * 33], sum[N * 33], num[N * 33][17], tot;ll ans[N];vector<pii> ask[N];void push_up(int rt) {sum[rt] = sum[ls[rt]] + sum[rs[rt]];for (int i = 0; i <= 16; i++) {num[rt][i] = num[ls[rt]][i] + num[rs[rt]][i];}
}void update(int &rt, int l, int r, int x) {if (!rt) {rt = ++tot;}if (l == r) {if (sum[rt]) {sum[rt] = 0;for (int i = 0; i <= 16; i++) {num[rt][i] = 0;}}else {sum[rt] = 1;for (int i = 0; i <= 16; i++) {num[rt][i] = x >> i & 1;}}return ;}int mid = l + r >> 1;if (x <= mid) {update(ls[rt], l, mid, x);}else {update(rs[rt], mid + 1, r, x);}push_up(rt);return ;
}int merge(int x, int y, int l, int r) {if (x == 0 || y == 0) {return x | y;}if (l == r) {for (int i = 0; i <= 16; i++) {num[x][i] ^= num[y][i];}sum[x] ^= sum[y];return x;}int mid = l + r >> 1;ls[x] = merge(ls[x], ls[y], l, mid);rs[x] = merge(rs[x], rs[y], mid + 1, r);push_up(x);return x;
}void add(int x, int y) {to[cnt] = y;nex[cnt] = head[x];head[x] = cnt++;
}ll query(int rt, int l, int r, int L, int R, int value) {if (!rt) {return 0;}if (l >= L && r <= R) {ll ans = 0;for (int i = 0; i <= 16; i++) {if (value >> i & 1) {ans += (1ll << i) * (sum[rt] - num[rt][i]);}else {ans += (1ll << i) * num[rt][i];}}return ans;}int mid = l + r >> 1;ll ans = 0;if (L <= mid) {ans += query(ls[rt], l, mid, L, R, value);}if (R > mid) {ans += query(rs[rt], mid + 1, r, L, R, value);}return ans;
}void dfs(int rt, int fa) {for (int i = head[rt]; i; i = nex[i]) {if (to[i] == fa) {continue;}dfs(to[i], rt);root[rt] = merge(root[rt], root[to[i]], 1, n);}update(root[rt], 1, n, value[rt]);for (auto it : ask[rt]) {if (it.second != n) {ans[it.first] = query(root[rt], 1, n, it.second + 1, n, it.second);}}
}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);for (int i = 1; i <= n; i++) {scanf("%d", &value[i]);}for (int i = 1; i < n; i++) {int x, y;scanf("%d %d", &x, &y);add(x, y);add(y, x);}for (int i = 1; i <= m; i++) {int u, x;scanf("%d %d", &u, &x);ask[u].push_back(make_pair(i, x));}dfs(1, 0);for (int i = 1; i <= m; i++) {printf("%lld\n", ans[i]);}return 0;
}

清明梦超能力者黄YY、异或树(线段树合并)相关推荐

  1. 清明梦超能力者黄YY[树链剖分+扫描线,线段树合并]

    清明梦超能力者黄YY 题目连接 https://www.nowcoder.com/acm/contest/206/I 暂时有两种做法. 算法一 涉及:树链剖分,扫描线 在一个线段的情况下,我们可以把一 ...

  2. 牛客国庆集训派对Day6 I 清明梦超能力者黄YY

    清明梦超能力者黄YY 传送门 题目: 黄YY是一个清明梦超能力者,同时也是一个记忆大师.他能够轻松控制自己在梦中的一切,在醒来之后还能清晰的记得梦中所有的细节,这让他的朋友们都十分羡慕. 又是一个晚上 ...

  3. 清明梦超能力者黄YY(idx数组)

    清明梦超能力者黄YY https://www.nowcoder.com/acm/contest/206/I 题目描述 黄YY是一个清明梦超能力者,同时也是一个记忆大师.他能够轻松控制自己在梦中的一切, ...

  4. 牛客国庆集训派对Day6 I.清明梦超能力者黄YY(树剖)

    题目:https://www.nowcoder.com/acm/contest/206/I 正难则反. 问你倒数第k次的颜色,正着来搞不定,那就转换成"倒着来的第k次". 使用树剖 ...

  5. 树套树-线段树套平衡树

    作用 线段树的作用是区间修改和查询,平衡树的作用是查询第k大,k的排名,前驱,后继.这两个结合起来,就变成了可以区间修改和查询第k大,k的排名,前驱,后继的数据结构:树套树-线段树套平衡树. 实现 先 ...

  6. P5127-子异和【线段树,树链剖分,位运算】

    正题 题目链接:https://www.luogu.org/problemnew/show/P5127 题目大意 定义一个序列的子异和为所有自己的异或和的和. 然后有点权的树,要求支持路径异或和路径求 ...

  7. 线段树 ---- 线段树维护线段相加+滑动变长窗口 2021牛客多校第7场 F xay loves trees

    题目大意: 给你两个大小相同的树但是形状不一定一样 叫你选出最大的子集,满足下面两个条件 在第一颗树上是一条链 在第二颗树上任意两个点都不是祖先关系 解题思路: 首先我们现在第二颗树上面把每个点的df ...

  8. BZOJ 3685: 普通van Emde Boas树( 线段树 )

    建颗权值线段树就行了...连离散化都不用... 没加读入优化就TLE, 加了就A掉了...而且还快了接近1/4.... ---------------------------------------- ...

  9. 2021CCPC(桂林) - Suffix Automaton(后缀树+线段树)

    题目链接:点击查看 题目大意:给出一个长度为 nnn 的字符串,再给出 qqq 次询问,每次询问需要输出本质不同第 kkk 小的子串的起止位置.如果有多个答案,输出起点最小的那个. 本题规定字符串大小 ...

最新文章

  1. UVA 617 - Nonstop Travel(数论+暴力枚举)
  2. 微信小程序(六) 文章详情静态页面detail
  3. P2756 魔术球问题 (网络流)
  4. android异步线程利用Handler将消息发送至UI线程
  5. 使用GZIP和压缩数据
  6. 最近很火的桌面小猫代打器 人性化UI设置界面
  7. pid调节软件_(四)差分底盘设计——5.底盘PID控制参数整定
  8. CSS按钮动画(二)
  9. Java俩个小球碰撞的代码_java实现小球碰撞功能
  10. springboot hikari数据库连接池死链 出现异常
  11. Tensorflow 中 learning rate decay 的奇技淫巧
  12. 未明学院学员报告:“民以食为天”?看了这份天猫超市数据分析报告你就知道了!
  13. 客户体验改善计划的用户注销通知导致服务器自动重启
  14. 用友BIP对接外部旺店通企业版奇门系统
  15. IDEA怎么查看现在的项目使用的JDK版本?
  16. Linux统计项目代码行数
  17. 银联在线网关支付,快速接入指南
  18. 17年1月9日,小程序来了。深度解析2017微信公开课
  19. 光流的计算(Lucas–Kanade method)
  20. YOLO系列总结:YOLOv1, YOLOv2, YOLOv3, YOLOv4, YOLOv5, YOLOX

热门文章

  1. java ecc 加密_java-信息安全(十一)-非对称加密算法002-ECC,签名003-ECDSA签名
  2. 华为怎么删除自带的音乐_原来华为手机相册隐藏剪辑功能!按下这个开关,还能制作音乐相册...
  3. 2020-11-09
  4. 如果觉得午休时间太短怎么办?
  5. 此内容过于真实,引起强烈舒适
  6. 娱乐项目和女朋友哪个重要?
  7. python中random什么意思_python中的random
  8. oracle取位置,获取oracle trace文件路径
  9. qbuttongroup如何都不选中_全程马拉松,半程马拉松该如何跑?很多人都不知道这些细节...
  10. python中getrandbits函数用法_python random - 刘江的python教程