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

题意:

给出n个数a[],然后给出m个问题Y,求a[]里面的X,有X xor Y最大。

字典树的作用之一----最大异或:

如果找一个数的最大异或,当然需要从高位到低位(已转化为二进制),尽可能不同。

  • 那么我们从高位到低位表示一个数X,并且存入字典树中,结尾节点记录X(不同的数结尾肯定不同)。
  • 对于询问,从高位到地位先尽可能从兄弟边走。

这可Trie树高度为32,复杂度在接受范围内。

#include<cstdio>
#include<cstdlib>
#include<iostream>
#include<cmath>
#include<algorithm>
#include<cstring>
using namespace std;
const int maxn=;
inline int read()
{
int res=;char x=getchar();while(x<''||x>'') x=getchar();
while(x>=''&&x<=''){ res=res*+x-'';x=getchar();}return res;
}
int q_pow(int a,int x){ int res=;while(x){ if(x&) res*=a; x>>=; a=a*a;} return res; }
struct TREE
{
int ch[maxn][],cnt,num[maxn];
void init() { memset(ch,,sizeof(ch));cnt=;}
void insert(int s[],int val)
{
int Now=;
for(int i=;i>=;i--){
if(!ch[Now][s[i]]) ch[Now][s[i]]=++cnt;
Now=ch[Now][s[i]];
} num[Now]=val;
}
int query(int s[])
{
int Now=;
for(int i=;i>=;i--){
if(ch[Now][s[i]^]) Now=ch[Now][s[i]^];
else Now=ch[Now][s[i]];
} return num[Now];
}
}Tree;
int main()
{
int T,n,m,x,tx,a[],Case=;
scanf("%d",&T);
while(T--){
printf("Case #%d:\n",++Case);
scanf("%d%d",&n,&m);
Tree.init();
for(int i=;i<=n;i++){
scanf("%d",&x);tx=x;
for(int j=;j<=;j++) {
a[j]=x%; x>>=;
} Tree.insert(a,tx);
}
for(int i=;i<=m;i++) {
scanf("%d",&x);
for(int j=;j<=;j++) {
a[j]=x%; x>>=;
} printf("%d\n",Tree.query(a));
}
}return ;
}

兄弟题目:Find MaxXorSum ,这个题求最大异或结果。

所以不需要记录来自于那个数字,记录下和即可。

#include<cstdio>
#include<cstdlib>
#include<iostream>
#include<cmath>
#include<algorithm>
#include<cstring>
using namespace std;
const int maxn=;
inline int read()
{
int res=;char x=getchar();while(x<''||x>'') x=getchar();
while(x>=''&&x<=''){ res=res*+x-'';x=getchar();}return res;
}
int q_pow(int a,int x){ int res=;while(x){ if(x&) res*=a; x>>=; a=a*a;} return res; }
struct TREE
{
int ch[maxn][],cnt;
void init() { memset(ch,,sizeof(ch));cnt=;}
void insert(int s[])
{
int Now=;
for(int i=;i>=;i--){
if(!ch[Now][s[i]]) ch[Now][s[i]]=++cnt;
Now=ch[Now][s[i]];
}
}
int query(int s[])
{
int Now=,res=;
for(int i=;i>=;i--){
if(ch[Now][s[i]^]) Now=ch[Now][s[i]^],res+=q_pow(,i);
else if(ch[Now][s[i]]) Now=ch[Now][s[i]];
else break;
} return res;
}
}Tree;
int main()
{
int T,n,i,j,a[],b[],tmp,ans;
scanf("%d",&T);
while(T--){
Tree.init(); ans=; scanf("%d",&n);
for(i=;i<=n;i++){
scanf("%d",&a[i]); tmp=a[i];
for(j=;j<=;j++) {
b[j]=tmp%; tmp>>=;
} Tree.insert(b);
}
for(i=;i<=n;i++) {
for(j=;j<=;j++) {
b[j]=a[i]%; a[i]>>=;
} ans=max(ans,Tree.query(b));
} printf("%d\n",ans);
}return ;
}

HDU4825 Xor Sum(字典树解决最大异或问题)的更多相关文章

  1. HDU--4825 Xor Sum (字典树)

    题目链接:HDU--4825 Xor Sum mmp sb字典树因为数组开的不够大一直wa 不是报的 re!!! 找了一下午bug 草 把每个数转化成二进制存字典树里面 然后尽量取与x这个位置上不相同 ...

  2. HDU4825 Xor Sum —— Trie树

    题目链接:https://vjudge.net/problem/HDU-4825 Xor Sum Time Limit: 2000/1000 MS (Java/Others)    Memory Li ...

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

    Xor Sum Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 132768/132768 K (Java/Others) Total ...

  4. HDU 4825 Xor Sum 字典树+位运算

    点击打开链接 Xor Sum Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 132768/132768 K (Java/Others) ...

  5. hdu 4825 xor sum(字典树+位运算)

    Xor Sum Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 132768/132768 K (Java/Others)Total S ...

  6. AcWing:143. 最大异或对(01字典树 + 位运算 + 异或性质)

    在给定的N个整数A1,A2……ANA1,A2……AN中选出两个进行xor(异或)运算,得到的结果最大是多少? 输入格式 第一行输入一个整数N. 第二行输入N个整数A1A1-ANAN. 输出格式 输出一 ...

  7. HDU-4825 Xor Sum,字典树好题!

    Xor Sum 一遍A了之后大呼一声好(keng)题!debug了两小时~~~~百度之星资格赛,可以. 题意:给你一个n个元素的数组,m次查询,每次输入一个数k要求从数组中找到一个数与k异或值最大,输 ...

  8. ☆ [HDU4825] Xor Sum「最大异或和(Trie树)」

    传送门:>Here< 题意:给出一个集合,包含N个数,每次询问给出一个数x,问x与集合中的一个数y异或得到最大值时,y是多少? 解题思路 由于N,M非常大,暴力显然不行.抓住重点是异或,所 ...

  9. [Hdu4825]Xor Sum(01字典树)

    Description Zeus 和 Prometheus 做了一个游戏,Prometheus 给 Zeus 一个集合,集合中包含了N个正整数,随后 Prometheus 将向 Zeus 发起M次询问 ...

随机推荐

  1. Swift oc 混编 - oc导入Swift协议

    (默认已经设置好桥接头文件)1.在Swift文件中写好协议2.oc类文件中导入:"项目名-swift.h"格式的文件 即:#include "项目名-swift.h&qu ...

  2. Android开发——内存优化 图片处理

    8.  用缓存避免内存泄漏 很常见的一个例子就是图片的三级缓存结构,分别为网络缓存,本地缓存以及内存缓存.在内存缓存逻辑类中,通常会定义这样的集合类. private HashMap<Strin ...

  3. win10 nginx + django +flup 配置

    1 安装 Nginx 官网下载,直接点exe启动即可 2 安装django pip install django 注意 新建立项目时 python *****/django-admin.py star ...

  4. Python 实现二维码生成和识别

    今天突然想给自己自己做个头像,然后还是二维码的形式,这样只要扫一扫就可以访问我的主页.然后就开始自己的苦逼之路... 其实实现二维码java,c#,C++等都可以实现:由于自己正在学python,所以 ...

  5. sql的一些知识_where

    简单相同查询,查询age=22的三列信息,并按照降序排列 ORDER BY weight DESC 此外,where还支持一下判断操作符 值得注意的是,如果比较的值为数值,则不需要加引号,如果是字符, ...

  6. project管理之makefile与自己主动创建makefile文件过程

    (风雪之隅 http://www.laruence.com/2009/11/18/1154.html) Linux Makefile自己主动编译和链接使用的环境 想知道到Linux Makefile系 ...

  7. android客户端向服务器端验证登陆方法的实现2

    一.在上一篇文章中,我只是提到了其中一种方法来实现登陆 大家可以参见: http://www.apkbus.com/android-45004-1-1.html      android获取web服务 ...

  8. Git bare repo with multiple branches

    http://stackoverflow.com/questions/9324762/git-bare-repo-with-multiple-branches Q: I want to make a ...

  9. Android.mk: recipe commences before first target. Stop.

    [GUIDE] Setup Android Development Environment on Ubuntu 14.04 Trusty Tahr Hi All, This originally wa ...

  10. Ubuntu Server 12.04 乱码

    sudo vim /etc/default/locale 将 下面的内容修改 LANG="zh_CN.UTF-8" LANGUAGE="zh_CN:zh" 修改 ...