最长异或路径

题目链接: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. Nacos(二)源码分析Nacos服务端注册示例流程

    上回我们讲解了客户端配置好nacos后,是如何进行注册到服务器的,那我们今天来讲解一下服务器端接收到注册实例请求后会做怎么样的处理. 首先还是把博主画的源码分析图例发一下,让大家对整个流程有一个大概的 ...

  2. 【Oracle】从删除的recyclebin中查看并恢复数据

    如果数据库中用了drop删除表,后面没有加上purge的话,会出现在oracle的回收机制中 dba_recyclebin可以查看当前删除的都是哪些 这个只是部分截图,可以看到删除的对象是什么,删除的 ...

  3. 【一天一个知识点系列】- Redis Cluser之数据分布

    数据分布 简述 分布式数据库首先要解决把整个数据集按照分区规则映射到多个节点的问题,即把数据集划分到多个节点上,每个节点负责整体数据的一个子集 分区及限制 分区规则 常见的分区规则 顺序分区 哈希分区 ...

  4. Promise.all()使用实例

    一.什么是Promise.all()? 在说这个之前要先说清楚promise.promise就是一个对象,专门用来处理异步操作的. 而Promise.all方法用于将多个 Promise 实例,包装成 ...

  5. 消息队列之kafka

    消息队列之activeMQ 消息队列之RabbitMQ 1.kafka介绍 kafka是由scala语言开发的一个多分区,多副本的并且居于zookeeper协调的分布式的发布-订阅消息系统.具有高吞吐 ...

  6. Python数据模型与Python对象模型

    数据模型==对象模型 Python官方文档说法是"Python数据模型",大多数Python书籍作者说法是"Python对象模型",它们是一个意思,表示&quo ...

  7. linux在终端中按下键盘立马反应

    想在终端中做个小应用,按下上下左右键能立刻作出反应. 测试程序见下: 1 #include <stdio.h> 2 #include <unistd.h> 3 #include ...

  8. 免安装的tomcat转服务

    一:确保tomcat 在点击bin\startup 文件可以正常启动访问: 二:本机安装有JDK: 三:本机环境变量配置:JAVA_HOME:C:\Java\jdk1.7.0_17; 四:本机Tomc ...

  9. Crunch

    Crunch 目录 1. 简介 2. 命令格式 3. options可选参数 3.1 -b number[type] 3.2 -c number 3.3 -d numbersymbol 3.4 -e ...

  10. 「笔记」AC 自动机

    目录 写在前面 定义 引入 构造 暴力 字典图优化 匹配 在线 离线 复杂度 完整代码 例题 P3796 [模板]AC 自动机(加强版) P3808 [模板]AC 自动机(简单版) 「JSOI2007 ...