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

清明梦超能力者黄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;
}

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

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

相关文章

net core WebApi——尝试企业微信来开发企业内部应用

前言这几天忙活着别的东西&#xff0c;耽误了很长时间&#xff0c;从文件操作完了之后就在考虑着下一步鼓捣点儿啥&#xff0c;因为最开始的业务开发就是企业微信相关的&#xff0c;这刚好来做个内部应用的小例子玩玩。企业微信前身是企业号&#xff0c;当时微信主推的还是公众…

2019-03-18-算法-进化(字符串中的第一个唯一字符)

题目描述 给定一个字符串&#xff0c;找到它的第一个不重复的字符&#xff0c;并返回它的索引。如果不存在&#xff0c;则返回 -1。 案例: s "leetcode" 返回 0.s "loveleetcode", 返回 2.注意事项&#xff1a;您可以假定该字符串只包含小写字母。 思…

P2619 [国家集训队]Tree I(WQS二分/带权二分/最小生成树)

P2619 [国家集训队]Tree I 给定一个n个点&#xff0c;m条边的无向图&#xff0c;每条边有一个颜色黑色或者白色&#xff0c;求解恰好有k条白色边的最小生成树。 那么看到恰好选择k个的最优性问题&#xff0c;我们可以利用WQS二分解决&#xff0c;实际上就是利用了对于每个选择…

E. Party Company(树上问题)

E. Party Company 容易发现这是一颗树形结构&#xff0c;根节点为111&#xff0c;并且有点权从根节点开始递减。 题目大意就是给定一个点u,l,ru, l, ru,l,r&#xff0c;对于于uuu在同一个连通块里&#xff0c;并且点权是在[l,r][l, r][l,r]之间的点答案贡献加一。 如果理解到…

微软发布了开发社区采用.NET Standard的最新信息

最近&#xff0c;微软发布了开发社区当前采用.NET Standard的最新信息。.NET Standard是API的正式规范&#xff0c;现有.NET实现在不同平台的是通用的&#xff08;从而允许跨平台开发&#xff09;。当前规范&#xff08;版本2.0&#xff09;在两年前发布&#xff0c;在.NET Cor…

2019-03-18-算法-进化(实现strStr())

题目描述 实现 strStr() 函数。 给定一个 haystack 字符串和一个 needle 字符串&#xff0c;在 haystack 字符串中找出 needle 字符串出现的第一个位置 (从0开始)。如果不存在&#xff0c;则返回 -1。 示例 1: 输入: haystack "hello", needle "ll" 输…

CF429E Points and Segments(欧拉回路)

CF429E Points and Segments 给定n 条线段[li,ri][l_i,r_i][li​,ri​] ,然后给这些线段红蓝染色&#xff0c;求最后直线上上任意一个点被蓝色及红色线段覆盖次数之差的绝对值不大于1 首先见到绝对值不大于1我们就容易想到欧拉回路&#xff0c;因为欧拉回路可以用来构造恰好相…

卓语言对泛型类的使用

上次发了中文编程语言卓语言《小卓.NET中文编程特点介绍》。这篇文章来看下卓语言对泛型类的使用。泛型是现代编程语言很重要的功能。C#语言可以完全定义和使用泛型类型。卓语言是面向广大非专业人员的&#xff0c;为了减低编程难度&#xff0c;没有实现定义泛型类型&#xff0…

快速傅里叶变换(完整推导过程 + 模板)

快速傅里叶变换 多项式表示 系数表示法&#xff1a; 一个nnn次多项式可以用n1n 1n1个系数表示出来&#xff1a;f(x)a0a1xa2x2⋯an−1xn−1anxnf(x) a_0 a_1 x a_2 x ^ 2 \dots a_{n - 1} x ^{n- 1} a_n x ^nf(x)a0​a1​xa2​x2⋯an−1​xn−1an​xn。 点值表示法&a…

2019-03-19-算法-进化(报数)

题目描述 报数序列是一个整数序列&#xff0c;按照其中的整数的顺序进行报数&#xff0c;得到下一个数。其前五项如下&#xff1a; 1. 1 2. 11 3. 21 4. 1211 5. 1112211 被读作 “one 1” (“一个一”) , 即 11。 11 被读作 “two 1s” (“两个一”&…

AT2675 [AGC018F] Two Trees(欧拉回路)

AT2675 [AGC018F] Two Trees 首先我们看到1或-1&#xff0c;那么就是限制差距在1以内&#xff0c;然后我们可以想到构造一些东西来满足这种东西&#xff0c;然后我们经常利用的就是欧拉回路。 首先这是两个树&#xff0c;然后我们可以根据儿子个数来判断当前点的奇偶性&#x…

.netcore 分布式事务CAP2.6之控制台使用

上一编.netcore 分布式事务CAP2.6 快速入门 讲了cap2.6的快速入门&#xff0c;这次我们来讲讲在控制台中如何使用cap2.6。因为cap2.6的内存模式目前已经可以使用了&#xff0c;相关组件已经更新&#xff0c;所以这次我们以简单的内存模式为例。1&#xff1a;创建项目创建一个名…

2019-03-18-算法-进化(有效的字母异位词)

给定两个字符串 s 和 t &#xff0c;编写一个函数来判断 t 是否是 s 的一个字母异位词。 示例 1: 输入: s "anagram", t "nagaram" 输出: true示例 2: 输入: s "rat", t "car" 输出: false说明: 你可以假设字符串只包含小写字母…

P6378 [PA2010] Riddle(2-sat/前后缀优化建图)

P6378 [PA2010] Riddle n个点m条边的无向图&#xff0c;分为k个部分&#xff0c;从每个部分选择恰好一个关键点&#xff0c;使得每条边至少有一个端点是关键点。 首先有这么多的限制&#xff0c;实际上就是一个选或者不选的问题&#xff0c;每条边的限制相当于一个不选就必须…

2019-03-18-算法-进化(删除链表的倒数第N个节点)

给定一个链表&#xff0c;删除链表的倒数第 n 个节点&#xff0c;并且返回链表的头结点。 示例&#xff1a; 给定一个链表: 1->2->3->4->5, 和 n 2.当删除了倒数第二个节点后&#xff0c;链表变为 1->2->3->5.说明&#xff1a; 给定的 n 保证是有效的。…

TestinPro应用与DevOps之路

文 | 中国农业银行软件研发中心 系统支持部 王晓昕 程伟静 胡莉莉Testin Pro&#xff08;云测平台&#xff09;是一款移动端自动化测试平台工具&#xff0c;帮助用户实现移动端测试自动化&#xff0c;是一套设备统一调配、软硬件一体化的移动端测试方案。Testin Pro具有在线录制…

多项式开根

多项式开根 给定多项式g(x)g(x)g(x)&#xff0c;求f(x)f(x)f(x)&#xff0c;满足f2(x)g(x)f ^ 2(x) g(x)f2(x)g(x)。 假设我们已经得到了g(x)g(x)g(x)&#xff0c;膜x⌈n2⌉x ^{\lceil \frac{n}{2} \rceil}x⌈2n​⌉下的根f0(x)f_0 (x)f0​(x)&#xff0c;要求膜xnx ^ nxn下…

通过Service访问应用 (2)

目录 通过NodePort Service在外部访问集群应用 通过LoadBalancer Service在外部访问集群应用 Microsoft SQL Server数据库部署 为了便于理解和学习&#xff0c;请先阅读上一篇《通过Service访问应用 &#xff08;1&#xff09;》再继续学习本篇内容。通过NodePort Service在外…

2019-03-18-算法-进化(反转链表)

题目描述 反转一个单链表。 示例: 输入: 1->2->3->4->5->NULL 输出: 5->4->3->2->1->NULL进阶: 你可以迭代或递归地反转链表。你能否用两种方法解决这道题&#xff1f; /*** 思路1&#xff1a;迭代法&#xff0c;直接依次反转链表* 时间复杂度…

分治FFT

分治FFT 考虑计算这么一个式子f(i)∑j1ifi−jg(j)f(i) \sum\limits_{j 1} ^{i} f_{i - j}g(j)f(i)j1∑i​fi−j​g(j)&#xff0c;给定g(x)g(x)g(x)&#xff0c;求f(x)f(x)f(x)&#xff0c;边界条件f(0)1f(0) 1f(0)1。 假设我们已经算出[l,mid][l, mid][l,mid]&#xff0c…