最长异或路径

题目链接:ybt高效进阶2-4-3 / luogu P4551

题目大意

给定一棵 n 个点的带权树,结点下标从 1 开始到 N。寻找树中找两个结点,求最长的异或路径。

异或路径指的是指两个结点之间唯一路径上的所有边权的异或。

思路

首先看到要异或的值最大,我们要想到可以用 Trie 树来贪心弄。

但是它好像不知道怎么弄,那我们先不管它。

那我们看到是一棵树,那我们可以试着统计 \(i\) 到根节点(我这里设是 \(1\))的异或路径的长度是多少。

那我们考虑能不能用这个表示出任意两个点之间的异或路径。

这里先给出结论,其实就是两个点到根节点的异或路径异或起来得出的值。

我们来证明:

分两种情况,分别是一个点在另一个点到根节点的路径上,要么就是两条路径是分开的,不会相交。

  1. 第一种,那我们可以知道一个点,就是一个值异或它自己就是 \(0\),就会消掉。那你想想,第一种情况时这个图:



    那 \(1\) 号点到根节点的异或路径就是 \(a\),\(2\) 号点到根节点的异或路径是 \(a\oplus b\),我们要的是 \(b\)。

    那你发现,把它们异或起来,就是 \(a\oplus a\oplus b=b\)。(两个 \(a\) 异或起来抵消掉了)
  2. 第二种,那我们可以画图。



    那 \(1\) 号点到根节点的异或路径就是 \(a\),\(2\) 号点到根节点的异或路径是 \(b\),我们要的是 \(a\oplus b\)。

    那你发现,把它们异或起来,就是 \(a\oplus b\)。

那你就可以一开始预处理出到根节点的异或路径,然后枚举两个点,然后算这两个点的异或路径,然后取最大值。

但是很明显这样是 \(O(n^2)\) 的,它会超时。

那我们就想一想有什么方法可以快速求最大值的。

想想我们之前一开始想用什么方法?

没错,就是 Trie 树。

我们可以把每个点到根节点的异或路径都放进 Trie 树里面构造。

然后每次枚举你要的异或路径的另一个点,然后跟 Trie 树里面的路径匹配找到最大值。

前面做过一题就是求这个最大值的,主要的就是用了贪心的思想。

从高位向低位枚举,然后如果有跟你这一位不同的就优先选,同时统计这一位异或之后是 \(1\) 对数的贡献。然后如果没有不同的,就看有没有相同的。

(因为毕竟你可以这一位相同,然后尽可能让后面更高的位不同,这样的贡献就更大)

那如果想相同不相同都没有,那就只能以当前的贡献退出了。

(如果想看之前的那一题可以点我查看,不过我只写在了 csdn,博客园里没有,因为比较简单)

然后对这些最大值选一个最大的,就是答案了。

代码

#include<cstdio>
#include<iostream> using namespace std; struct node {
int x, to, nxt;
}e[200001];
struct Tree {
int son[2];
}trie[1000001];
int n, x, y, z, le[100001], KK, go, KKK, ans; void add(int x, int y, int z) {//邻接表
e[++KK] = {z, y, le[x]}; le[x] = KK;
e[++KK] = {z, x, le[y]}; le[y] = KK;
} void build(int num) {//Trie树建树
int now = 0;
for (int i = 31; i >= 0; i--) {
go = num >> i & 1;
if (!trie[now].son[go]) trie[now].son[go] = ++KKK;
now = trie[now].son[go];
}
} int find(int num) {
int now = 0, re = 0;
for (int i = 31; i >= 0; i--) {//从高位到低位贪心看
go = num >> i & 1;
if (trie[now].son[go ^ 1]) {//先看能不能有这一位不同
now = trie[now].son[go ^ 1];
re |= 1 << i;
}
else if (trie[now].son[go]) now = trie[now].son[go];//只能相同
else return re;//都没有,就只能退出了
}
return re;
} void dfs1(int now, int father, int num) {//建出从根节点到 i 点的异或路径构成的 Trie 数
build(num);
for (int i = le[now]; i; i = e[i].nxt)
if (e[i].to != father)
dfs1(e[i].to, now, num ^ e[i].x);
} void dfs2(int now, int father, int num) {//得出与现在的路径异或能得到的最大值
ans = max(ans, find(num));
for (int i = le[now]; i; i = e[i].nxt)
if (e[i].to != father)
dfs2(e[i].to, now, num ^ e[i].x);
} int main() {
scanf("%d", &n);
for (int i = 1; i < n; i++) {
scanf("%d %d %d", &x, &y, &z);
add(x, y, z);
} dfs1(1, 0, 0); dfs2(1, 0, 0); printf("%d", ans); return 0;
}

【ybt高效进阶2-4-3】【luogu P4551】最长异或路径的更多相关文章

  1. [luogu] P4551 最长异或路径(贪心)

    P4551 最长异或路径 题目描述 给定一棵\(n\)个点的带权树,结点下标从\(1\)开始到\(N\).寻找树中找两个结点,求最长的异或路径. 异或路径指的是指两个结点之间唯一路径上的所有边权的异或 ...

  2. Luogu P4551 最长异或路径

    题目链接 \(Click\) \(Here\) \(01Trie\)好题裸题. 取节点\(1\)为根节点,向下扫每一个点从根节点到它路径上的异或和,我们可以得到一个\(sumx[u]\). 现在路径异 ...

  3. Luogu P4551 最长异或路径 01trie

    做一个树上前缀异或和,然后把前缀和插到$01trie$里,然后再对每一个前缀异或和整个查一遍,在树上从高位向低位贪心,按位优先选择不同的,就能贪出最大的答案. #include<cstdio&g ...

  4. 洛谷P4551 最长异或路径

    传送门:https://www.luogu.org/problem/show?pid=4551 在看这道题之前,我们应懂这道题怎么做:给定n个数和一个数m,求m和哪一个数的异或值最大. 一种很不错的做 ...

  5. 2018.10.26 洛谷P4551 最长异或路径(01trie)

    传送门 直接把每个点到根节点的异或距离插入01trie. 然后枚举每个点在01trie上匹配来更新答案就行了. 代码: #include<iostream> #include<cst ...

  6. P4551 最长异或路径

    题目描述 给定一棵 nnn 个点的带权树,结点下标从 111 开始到 NNN .寻找树中找两个结点,求最长的异或路径. 异或路径指的是指两个结点之间唯一路径上的所有边权的异或. 输入输出格式 输入格式 ...

  7. 洛谷 P4551 最长异或路径

    题目描述 给定一棵 nn 个点的带权树,结点下标从 11 开始到 NN .寻找树中找两个结点,求最长的异或路径. 异或路径指的是指两个结点之间唯一路径上的所有节点权值的异或. 输入输出格式 输入格式: ...

  8. P4551 最长异或路径 (01字典树,异或前缀和)

    题目描述 给定一棵 n 个点的带权树,结点下标从 1 开始到 N .寻找树中找两个结点,求最长的异或路径. 异或路径指的是指两个结点之间唯一路径上的所有边权的异或. 输入输出格式 输入格式: 第一行一 ...

  9. 洛谷【P4551】最长异或路径

    浅谈\(Trie\):https://www.cnblogs.com/AKMer/p/10444829.html 题目传送门:https://www.luogu.org/problemnew/show ...

随机推荐

  1. 【Oracle】CBO优化详解

    SQL优化是数据优化的重要方面,本文将分析Oracle自身的CBO优化,即基于成本的优化方法.Oracle为了自动的优化sql语句需要各种统计数据作为优化基础.外面会通过sql的追踪来分析sql的执行 ...

  2. 最新详解android自动化无障碍服务accessibilityservice以及高版本问题_1_如何开启获得无障碍

    前言 无障碍服务accessibilityservice是什么 简单来说 无障碍服务就是一个为残障人士 尤其是视觉障碍人士提供的一个帮助服务.具体就是可以识别控件 文字 可以配合语音助手 操作和 使用 ...

  3. ctfhub技能树—web前置技能—http协议—302跳转

    开启靶机 打开环境,查看显示 点击Give me Flag后发生跳转 根据题目提示为HTTP临时重定向 简单记录一下HTTP临时重定向是什么 HTTP重定向:服务器无法处理浏览器发送过来的请求(req ...

  4. window安装nvm

    先说一下背景,最近做的两个项目一个是祖传angularjs1.X版本另一个是react hooks结合tailwindcss,前者angularjs的node版本比较低,而tailwindcss的no ...

  5. php 换行符

    PHP 中换行可以用 PHP_EOL 来替代,以提高代码的源代码级可移植性: unix系列用 \n windows系列用 \r\n mac用 \r 总结:在一些大文本域中换行的文本可以用这个来进行切割 ...

  6. 二十七:XSS跨站之代码及httponly绕过

    httponly:如果给某个 cookie 设置了 httpOnly 属性,则无法通过 JS 脚本 读取到该 cookie 的信息,但还Application 中手动修改 cookie,所以只是在一定 ...

  7. 自翻------Office 2013 RT 使用说明

    Office Home and Student 2013 RT Preview的更新 介绍 Microsoft已发布Microsoft Office Home and Student 2013 RT ...

  8. 一个简单的IM聊天程序Pie IM(以后会更新)

    这个程序用多线程,实现设备之间的聊天,支持win10通知,欢迎下载 依赖的第三方库 win10toast 代码 将以下代码写入任意.py文件 1 print('Welcome to use Pie I ...

  9. Hive常用日期格式转换

    固定日期转换成时间戳 select unix_timestamp('2016-08-16','yyyy-MM-dd') --1471276800 select unix_timestamp('2016 ...

  10. Qedis实现

    对比redis的Qedis 实现在github 和 实验楼都有资料