算法刷题:最大异或对(Trie树扩展)、食物链(并查集扩展)

目录

  • 引言
  • 一、最大异或对(Trie树扩展)
    • 1.题目描述
    • 2.解题思路
    • 3.代码实现
    • 4.测试
  • 二、食物链(并查集扩展)
    • 1.题目描述
    • 2.解题思路
    • 3.代码实现
    • 4.测试

引言

这两个扩展题能够让我们更加的熟悉Trie树和并查集的使用,这两道题可以说是比较难的了,所以说还是好好对待吧。

一、最大异或对(Trie树扩展)

1.题目描述

在给定的 N 个整数 A1,A2……AN 中选出两个进行 xor(异或)运算,得到的结果最大是多少?输入格式
第一行输入一个整数 N。
第二行输入 N 个整数 A1~AN。输出格式
输出一个整数表示答案。数据范围
1≤N≤105,0≤Ai<231输入样例:
3
1 2 3输出样例:
3

2.解题思路

  • 首先最开始当然是用暴力做一下,如下所示,很明显是O(n^2)的一个时间复杂度,然后这个数据是10 ^ 5,那就是执行10 ^ 10次,然后一般C++的时间范围在10 ^ 9内一般都是可以过的,这个当然就TLE了。
int res = 0;
for (int i = 0; i < n; ++i)
{for (int j = 0; j < i; ++j){res = max(res, a[i] ^ a[j]);}
}
  • 然后就想怎么优化了,我们注意第二层循环是查找当然数中与a[i]异或的最大值,我们发现数据是小于 2 ^ 31 - 1的,也就是最多31位,那么是最大的,肯定先从最高位找有没有一个数跟a[i]的最高位不一样,如果有那就把他们集合到一块,没有那就挑相同的,然后找第29位有没有与a[i]第29位不同位的树,如果没有那就找相同的,然后就这样找到第0位,肯定就是与a[i]异或最大的数了,这也是一种贪心的思想,具体图解看如下的例子:
    如果右边是已经有的Trie树了,我们要找与110异或最大的,那么最高位是1,我们就要找0,然后有0,然后看下一位是1,我们就要找0,没有0,那就只能是1了,然后下一位是0,我们要找1,那就找到了,所以最终与110异或最大的数是011
    在这里插入图片描述

3.代码实现

#include <cstdio>
#include <iostream>using namespace std;const int N = 1e5+10, M = 31 * N;  // 最大有M个结点int a[N];
int son[M][2], idx;  // M个结点,每个结点两个儿子0,1
int n;void insert(int x)
{int p = 0;for(int i = 30; i >= 0; --i){int u = x >> i & 1;if(!son[p][u]) son[p][u] = ++idx;p = son[p][u];}
}int query(int x)
{int p = 0, res = 0;for(int i = 30; i >= 0; --i){int u = x >> i & 1;if(son[p][!u])  // 找不同的{res = res * 2 + !u;p = son[p][!u];}else  // 如果不存在,那只能退而求其次{res = res * 2 + u;p = son[p][u];}}return res;
}int main()
{scanf("%d", &n);for(int i = 0; i < n; ++i) scanf("%d", &a[i]);int res = 0;for(int i = 0; i < n; ++i){insert(a[i]);int t = query(a[i]);  //查询与a[i]异或最大的数res = max(res, a[i] ^ t);}printf("%d\n", res);return 0;
}

4.测试

在这里插入图片描述

在这里插入图片描述

二、食物链(并查集扩展)

1.题目描述

动物王国中有三类动物 A,B,C,这三类动物的食物链构成了有趣的环形。A 吃 B,B 吃 C,C 吃 A。现有 N 个动物,以 1∼N 编号。每个动物都是 A,B,C 中的一种,但是我们并不知道它到底是哪一种。有人用两种说法对这 N 个动物所构成的食物链关系进行描述:
第一种说法是 1 X Y,表示 X 和 Y 是同类。
第二种说法是 2 X Y,表示 X 吃 Y。此人对 N 个动物,用上述两种说法,一句接一句地说出 K 句话,这 K 句话有的是真的,有的是假的。当一句话满足下列三条之一时,这句话就是假话,否则就是真话。
当前的话与前面的某些真的话冲突,就是假话;
当前的话中 X 或 Y 比 N 大,就是假话;
当前的话表示 X 吃 X,就是假话。你的任务是根据给定的 N 和 K 句话,输出假话的总数。输入格式
第一行是两个整数 N 和 K,以一个空格分隔。
以下 K 行每行是三个正整数 D,X,Y,两数之间用一个空格隔开,其中 D 表示说法的种类。
若 D=1,则表示 X 和 Y 是同类。
若 D=2,则表示 X 吃 Y。输出格式
只有一个整数,表示假话的数目。数据范围
1≤N≤50000 ,0≤K≤100000输入样例:
100 7
1 101 1 
2 1 2
2 2 3 
2 3 3 
1 1 3 
2 3 1 
1 5 5输出样例:
3

2.解题思路

这道题是要输出谎话的次数,然后x,y > n,就不说了很好判断,然后其实就是判断是不是同类,是不是能吃的关系
思路就是如下图的关系,是不是同类取决于到根结点的距离是否相同,吃与被吃的关系却决于到根结点的距离,当然这些距离都要 mod 3

3.代码实现

关于最后 d[px] 的推导我画了张图,虽然很丑,但是关系描述的还是很清晰的,然后注释写的也比较详细,不懂得可以多看看
在这里插入图片描述

在这里插入图片描述

#include <cstdio>
#include <iostream>using namespace std;const int N = 5e5+10;int p[N], d[N];  // d[i]是i号结点到根结点的距离
int n, m;int find(int x)
{if(x != p[x]) {int t = find(p[x]);d[x] += d[p[x]];  // 必须先执行find操作,将d[p[x]是真正的到根结点的距离,而不是到上一个结点的距离p[x] = t;  // 路径压缩}return p[x];
}int main()
{scanf("%d%d", &n, &m);for(int i = 1; i <= n; ++i) p[i] = i;int res = 0;while(m--){int t, x, y;scanf("%d%d%d", &t, &x, &y);int px = find(x), py = find(y);if(x > n || y > n) res++;else{if(t == 1)  // 同类{if(px == py && (d[x] - d[y]) % 3) res++;  //如果是一个集合的,并且到根结点的距离不同,说明不是一个类的else if(px != py)  // 刚开始还没插进集合里{p[px] = py;  // 让x的根结点的父亲为y的根结点d[px] = d[y] - d[x];  // 只用将x根结点到y根结点的距离一改,之后的话,会通过上面的find函数自动修改的}}else  // x吃y{int link = 1;  // 这里给出2也是可以的,x比y多1或者2 都表示x吃yif(px == py && (d[x] - d[y] - link) % 3) res++;else if(px != py){p[px] = py;d[px] = d[y] - d[x] + link;}}}}printf("%d\n", res);return 0;
}

4.测试

在这里插入图片描述

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

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

相关文章

Linux:apache优化(2)—— 网页传输压缩

网页传输压缩 客户端在请求httpd服务器数据&#xff0c;httpd服务器在返回数据包给客户端时&#xff0c;先对返回的数据进行压缩&#xff0c;压缩之后再传输 作用&#xff1a;配置 Apache 的网页压缩功能&#xff0c;是使用 Gzip 压缩算法来对 Apache 服务器发布的网页内容进行…

【Spring】spring的容器创建

目录 控制反转IOC 依赖注入DI 创建spring的容器方式&#xff1a; 思考&#xff1a; spring整合Junit4 控制反转IOC 把对象的创建和对象之间的调用过程&#xff0c;交给Spring管理&#xff0c;IOC是容器&#xff0c;是思想。&#xff01;&#xff01;&#xff01; 依赖注入…

关于ISI、SCI、EI、IEEE、Elsevier、Springer, etc的理解

文章目录 0、一文搞定 学术文章发表 基本概念1、关于会议组织者、出版商、科研工作者的关系1.1 SCI到底是什么1.2 EI到底是什么1.3 EI-SCI-收录-检索-出版商-会议组织者关系 2、关于ISI、SCI 、汤森路透、路透社的关系2.1 ISI web of knowledge2.2 Master Journal List2.3 Jour…

按摩上门预约小程序源码系统 开发组合:PHP+MySQL 附带完整的搭建教程

现代生活节奏的加快&#xff0c;人们越来越注重健康与放松。按摩作为传统的舒缓方式&#xff0c;市场需求逐年上升。然而&#xff0c;传统的按摩服务预约方式较为繁琐&#xff0c;用户需拨打热线电话或前往实体店进行预约&#xff0c;这无疑增加了用户的操作成本。因此&#xf…

vscode 支持c,c++编译调试方法

概述&#xff1a;tasks.jason launch.json settings.json一定要有&#xff0c;没有就别想跑。还有就是c 和c配置有区别&#xff0c;切记&#xff0c;下文有说 1.安装扩展插件。 2.安装编译器&#xff0c;gcc.我用的是x86_64-8.1.0-release-win32-seh-rt_v6-rev0.7z &#xf…

nginx日志常见报错解决

目录 一&#xff1a;报错 二&#xff1a;php查看后台内容有的栏目出现502&#xff1f; 三&#xff1a;413 Request Entity Too Large? 四&#xff1a;Request Header Or Cookie Too Large 400 一&#xff1a;报错 upstream prematurely closed connection while reading r…

35.搜索插入位置

给定一个排序数组和一个目标值&#xff0c;在数组中找到目标值&#xff0c;并返回其索引。如果目标值不存在于数组中&#xff0c;返回它将会被按顺序插入的位置。 请必须使用时间复杂度为 O(log n) 的算法。 示例 1: 输入: nums [1,3,5,6], target 5 输出: 2示例 2: 输入:…

docker-compose Install TeamCity

前言 TeamCity 是一个通用的 CI/CD 软件平台,可实现灵活的工作流程、协作和开发实践。允许在您的 DevOps 流程中成功实现持续集成、持续交付和持续部署。 系统支持 docker download TeamCity TeamCity 文档参考项目离线包百度网盘获取

117基于matlab的短时傅里叶变换(STFT)、小波变换(WT)、同步压缩变换(SST)、瞬态提取变换(TET)进行时频分析

基于matlab的短时傅里叶变换&#xff08;STFT&#xff09;、小波变换&#xff08;WT&#xff09;、同步压缩变换&#xff08;SST&#xff09;、瞬态提取变换&#xff08;TET&#xff09;进行时频分析。程序已调通&#xff0c;可直接运行。 117时频分析短时傅里叶变换 (xiaohong…

python基础-01

文章目录 前言一、python中的注释二、变量的数据类型1.Number&#xff08;数字&#xff09;2.Boolean&#xff08;布尔类型&#xff09;—— True 和 False3.String&#xff08;字符串&#xff09;4.List&#xff08;列表&#xff09;5.Tuple&#xff08;元组&#xff09;6.Dic…

javaWeb学生信息管理系统2

一、学生信息管理系统SIMS 一款基于纯Servlet技术开发的学生信息管理系统&#xff08;SIMS&#xff09;&#xff0c;在设计中没有采用SpringMVC和Spring Boot等框架。系统完全依赖于Servlet来处理HTTP请求和管理学生信息&#xff0c;实现了信息的有效存储、检索和更新&#xf…

【 C语言 】| C程序百例 - 绘制余弦曲线

【 C语言 】| C程序百例 - 绘制余弦曲线 时间&#xff1a;2023年12月29日12:56:29 文章目录 【 C语言 】| C程序百例 - 绘制余弦曲线1.要求2.问题分析与算法设计3.程序3-1.源码3-2.makefile 4.运行 1.要求 在屏幕上用"*"显示0~360的余弦曲线cos(x)曲线。 2.问题分析与…