2014百度之星第三题Xor Sum(字典树+异或运算)「建议收藏」

2014百度之星第三题Xor Sum(字典树+异或运算)

大家好,又见面了,我是全栈君。

Xor Sum
Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 132768/132768 K (Java/Others)
Total Submission(s): 4445    Accepted Submission(s): 652

Problem Description
Zeus 和 Prometheus 做了一个游戏。Prometheus 给 Zeus 一个集合,集合中包括了N个正整数。随后 Prometheus 将向 Zeus 发起M次询问,每次询问中包括一个正整数 S 。之后 Zeus 须要在集合其中找出一个正整数 K ,使得 K 与 S 的异或结果最大。

Prometheus 为了让 Zeus 看到人类的伟大。随即允许 Zeus 能够向人类求助。你能证明人类的智慧么?
 
Input
输入包括若干组測试数据,每组測试数据包括若干行。输入的第一行是一个整数T(T < 10),表示共同拥有T组数据。每组数据的第一行输入两个正整数N,M(<1=N,M<=100000),接下来一行,包括N个正整数,代表 Zeus 的获得的集合,之后M行,每行一个正整数S。代表 Prometheus 询问的正整数。全部正整数均不超过2^32。

Output
对于每组数据,首先须要输出单独一行”Case #?

:”,当中问号处应填入当前的数据组数。组数从1開始计算。

对于每一个询问,输出一个正整数K,使得K与S异或值最大。
 
Sample Input
2
3 2
3 4 5
1
5
4 1
4 6 5 6
3

Sample Output
Case #1:
4
3
Case #2:
4

 

题解

    本题数据量非常大,感觉直接暴力时间复杂度O(n*m)会超时。只是想到去年网赛有道题数据量也非常大直接暴力居然过了于是直接敲了敲,终于还是TLE了。

想到位操作运算符的通用思想,逐位处理,相应本题还是比較easy想到字典树的。Memory Limit: 132768/132768 K (Java/Others),提示能够尝试牺牲空间换取时间。直接字典树空间消耗为2^32,但本题数据相对此规模还是比較小的。一共n个数,总共32*n位。所以空间复杂度为O(n)。能够接受。先用原序列中的数按位建立字典树。然后将查找过程中用待询问数与0xffffffff异或XOR来在字典树上跑。终于找到的即为最大的。

如果按查询的XOR的某个分支不存在。则想还有一分支进行。这样答案可能变小,可是正确的。总的时间复杂度O(m*log(32))。常数能够不计。

#include<iostream>
#include<cstring>
#include<cstdio>
using namespace std;

const int MAXN=100000*32+100;
__int64 tree[MAXN][2];
__int64 node[MAXN];
__int64 tot;

void insert(__int64 a)
{
	__int64 j;
	int bit,dep=0;
	for(j=31;j>=0;j--)
	{
		bit=((0x1<<j)&a)?0x1:0x0;
		if(tree[dep][bit]==0)
		{
			tree[dep][bit]=tot++;
			memset(tree[tree[dep][bit]], 0, sizeof(tree[tree[dep][bit]]));
		}
		dep=tree[dep][bit];
	}
	node[dep]=a;
}

__int64 Solve(__int64 val)
{
	__int64 j,bit;
	int dep=0;
	for(j=31;j>=0;j--)
	{
		bit=((0x1<<j)&val)?0x0:0x1;
		if(tree[dep][bit])dep=tree[dep][bit];
		else dep=tree[dep][bit^1];
	}
	return node[dep];
}

int main()
{
	int cas;
	__int64 n,m,j,s,a;
	int tag=0;
	cin>>cas;
	while(cas--)
	{
		tot=1;
		scanf("%I64d%I64d",&n,&m);
		memset(tree[0],0,sizeof(tree[0]));
		while(n--)
		{
			scanf("%I64d",&a);
			
			insert(a);
		}
		printf("Case #%d:\n",++tag);
		while(m--)
		{
			scanf("%I64d",&s);
			printf("%I64d\n",Solve(s));
		}
	}
	return 0;
}

 

 


 

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/116591.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • mipi 协议简介[通俗易懂]

    mipi 协议简介[通俗易懂]转载于:https://blog.csdn.net/g_salamander/article/details/9163455以下是最近几个月在调试 MIPIDSI/CSI 的一些经验总结,因为协议有专门的文档,所以这里就记录一些常用知识点:一、D-PHY1、传输模式LP(Low-Power)模式:用于传输控制信号,最高速率10MHzHS(High-Speed)模式:用于高速传输数据,速…

    2022年4月30日
    239
  • 猴子摘香蕉问题c语言_c语言人工智能算法

    猴子摘香蕉问题c语言_c语言人工智能算法问题说明:房间内有一只猴子,一个箱子和一个挂在天花板上的香蕉。三者的位置如下图所示:初始状态:三者在输入的初始位置,猴子手上无香蕉,猴子不在箱子上。目标状态:三者均在香蕉对应的位置,猴子手上有香蕉,且在箱子上。实现步骤:猴子走到箱子处猴子将箱子推到香蕉处猴子爬上箱子猴子摘香蕉程序内容:本程序主要实现猴子摘香蕉的过程,即从初始状态到目标状态。程序运行后,根据用户输入的三者的位置,按照实现步骤更新每一过程后的状态变量,并将过程输出。本程序使用以下函数:main():主函数

    2022年9月26日
    3
  • LHS和RHS理解

    LHS和RHS理解最近在重学前端 遇到 LHS 和 RHS 两个名词 这里记录下 方便深入理解两个概念见名知意 L 和 R 的含义 它们分别代表左侧和右侧 这里举一个简单的例子 console log a 在这段代码中 a 就是进行 RHS 查询 因为我们并没有对 a 进行赋值操作 而是直接引用了 a 我们需要查找并拿到 a 的值才能传递给 console log 如果 a 2 这里对 a 的引用则是 LHS 引用 LHS

    2025年8月23日
    3
  • 怎么理解泊松分布_泊松分布公式

    怎么理解泊松分布_泊松分布公式1甜在心馒头店公司楼下有家馒头店:每天早上六点到十点营业,生意挺好,就是发愁一个事情,应该准备多少个馒头才能既不浪费又能充分供应?老板统计了一周每日卖出的馒头(为了方便计算和讲解,缩小了数据):均值为:按道理讲均值是不错的选择(参见如何理解最小二乘法?),但是如果每天准备5个馒头的话,从统计表来看,至少有两天不够卖,的时间不够卖:你“甜在心馒头店”又不是…

    2022年8月30日
    2
  • Eurake分区理解

    Eurake分区理解Eurake分区理解大型项目如果存在多个机房,例如北京机房,上海机房,杭州机房等,上千个服务注册在Eurake上面,我们的事例也分别部署在各个区域。这时候,由于机房存在不同的区域,北京的一个服务如果调用上海的一个服务,就可能发生延迟,服务的响应速度也会慢很多,这时候,我们可能期望,北京的服务生产者调用北京的服务消费着,这该怎么去操作?Eurake其实有个分区功能,什么是分区,就是北京有一个注册…

    2022年6月12日
    37
  • sobel算子实现

    sobel算子实现原理:实现://阶乘intfactorial(intn){ intfac=1; if(n==0) returnfac; for(inti=1;i<=n;++i) fac*=i; returnfac;}//获得Sobel平滑算子MatgetSobelSmooth(intsize){ intn=size-1; MatSobelSmoothoper=Mat::zeros(size,1,CV_32F); fo

    2022年7月14日
    17

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号