题目链接

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

分析:

首先我们将数字转换为33位的二进制,不足33位的用0补充,然后根据每个数字的二进制位建立字典树。(需要注意的一点就是我们把一个数字转换为二进制后肯定是倒序的,但是我们建树的时候需要按照正序的来建树,所以建树的时候是要将求出的二进制位倒序建树)。

接下来对于每个查询,同样处理成33位二进制字符串,因为要求的是异或值最大,我们可以将每个位置的0存成1,1存成0,不足33位的全部用1填充,所以我们在查询的时候,如果当前位置已经建立过字典树,也就相当于与原来的值不相同,接着按照这个位置往下走(这样走下去的值是最大的),如果当前位置没有建立过字典树(也就是说与该二进制维相反的原来没有数字,那就只能走与它相同的了),也就相当于当前位置为0的话就走1,为1的话就走0。

代码:

  1. #include<stdio.h>
  2. #include<iostream>
  3. #include<string.h>
  4. using namespace std;
  5. int num[55];
  6. int val[6000000];
  7. int tree[6000000][2];
  8. int flag;
  9. void init()
  10. {
  11. memset(tree,0,sizeof(tree));
  12. memset(val,0,sizeof(val));
  13. flag=1;
  14. }
  15. void build(int a)
  16. {
  17. int u=0;
  18. for(int i=33;i>=1;i--)
  19. {
  20. if(tree[u][num[i]]==0)//没有值,就构建值
  21. {
  22. tree[u][num[i]]=flag++;
  23. }
  24. u=tree[u][num[i]];
  25. }
  26. val[u]=a;
  27. }
  28. int query()
  29. {
  30. int u=0;
  31. for(int i=33;i>=1;i--)
  32. {
  33. if(tree[u][num[i]]==0)//相当于与该位置的值相反的字典树中没有,那就只能按照相同的往下走了
  34. u=tree[u][num[i]^1];
  35. else
  36. u=tree[u][num[i]];//字典树中由值,在高位就异或后为1了,接着走异或值会更大
  37. }
  38. return val[u];
  39. }
  40. int main()
  41. {
  42. int T,n,m,a,cnt;
  43. scanf("%d",&T);
  44. for(int t=1;t<=T;t++)
  45. {
  46. init();
  47. printf("Case #%d:\n",t);
  48. scanf("%d%d",&n,&m);
  49. for(int i=0;i<n;i++)
  50. {
  51. memset(num,0,sizeof(num));
  52. scanf("%d",&a);
  53. cnt=0;
  54. int b=a;
  55. while(b)//这里还是按照原本的二进制位置存储
  56. {
  57. cnt++;
  58. if(b&1) num[cnt]=1;
  59. else num[cnt]=0;
  60. b>>=1;
  61. }
  62. build(a);
  63. }
  64. for(int i=0;i<m;i++)
  65. {
  66. scanf("%d",&a);
  67. cnt=0;
  68. while(a)//注意这里,是将每一个位置的二进制树都存反了(1存成0,0存成1)
  69. {
  70. cnt++;
  71. if(a&1) num[cnt]=0;
  72. else num[cnt]=1;
  73. a>>=1;
  74. }
  75. for(int j=cnt+1;j<=33;j++)
  76. num[j]=1;
  77. printf("%d\n",query());
  78. }
  79. }
  80. return 0;
  81. }

HDU 4825 Xor Sum (裸字典树+二进制异或)的更多相关文章

  1. HDU 4825 Xor Sum(字典树)

    嗯... 题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4825 这道题更明确的说是一道01字典树,如果ch[u][id^1]有值,那么就向下继续查找/ ...

  2. HDU 4825 Xor Sum (trie树处理异或)

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

  3. hdu 4825 Xor Sum(trie+贪心)

    hdu 4825 Xor Sum(trie+贪心) 刚刚补了前天的CF的D题再做这题感觉轻松了许多.简直一个模子啊...跑树上异或x最大值.贪心地让某位的值与x对应位的值不同即可. #include ...

  4. HDU 4825 Xor Sum(二进制的字典树,数组模拟)

    题目 //居然可以用字典树...//用cin,cout等输入输出会超时 //这是从别处复制来的 #include<cstdio> #include<algorithm> #in ...

  5. HDU 4825 Xor Sum(经典01字典树+贪心)

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

  6. hdu 4825 Xor Sum(01字典树模版题)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4825 题解:一到01字典树的模版题,01字典树就是就是将一些树用二进制放到一个树上这样可以方便对整体异 ...

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

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

  8. HDU 4825 Xor Sum(01字典树入门题)

    http://acm.hdu.edu.cn/showproblem.php?pid=4825 题意: 给出一些数,然后给出多个询问,每个询问要从之前给出的数中选择异或起来后值最大的数. 思路:将给出的 ...

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

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

随机推荐

  1. window.location.hash 使用

    [转]http://www.cnblogs.com/nifengs/p/5104763.html location是javascript里边管理地址栏的内置对象,比如location.href就管理页 ...

  2. False Ordering LightOJ - 1109(暴力。。唉,。又是一个水题。。)

    We define b is a Divisor of a number a if a is divisible by b. So, the divisors of 12 are 1, 2, 3, 4 ...

  3. 查看ubuntu版本号

    No.1 cat /etc/issue No.2 cat /proc/version No.3 uname -a No.4 lsb_release -a

  4. 【BZOJ 1098】办公楼(补图连通块个数,Bfs)

    补图连通块个数这大概是一个套路吧,我之前没有见到过,想了好久都没有想出来QaQ 事实上这个做法本身就是一个朴素算法,但进行巧妙的实现,就可以分析出它的上界不会超过 $O(n + m)$. 接下来介绍一 ...

  5. 【bzoj3926】 Zjoi2015—诸神眷顾的幻想乡

    http://www.lydsy.com/JudgeOnline/problem.php?id=3926 (题目链接) 题意 给出一棵树,每个节点有一个编号,范围在${[0,9]}$.一个序列是指树上 ...

  6. 单点登录(十七)----cas4.2.x登录mongodb验证方式成功后返回更多信息更多属性到客户端

    我们在之前已经完成了cas4.2.x登录使用mongodb验证方式登录成功了.也解决了登录名中使用中文乱码的问题. 单点登录(十三)-----实战-----cas4.2.X登录启用mongodb验证方 ...

  7. 单点登录(三)-----实战-----cas server 源码下载和部署

    我们在上一篇文章中使用的是4.0版本的cas,4.0版本的有发布好的war包可以直接使用,那如果我们要使用更新的版本怎么办呢? 就需要下载源码自己编辑打包了. 步骤如下: 版本选择 我们在cas的gi ...

  8. Java考试题之七

    QUESTION 150 Click the Exhibit button. Given: ClassA a = new ClassA(); a.methodA(); What is the resu ...

  9. Java EE之通过表单上传文件

    public class Ticket { private String customerName; private String subject; private String body; priv ...

  10. java 批量文件后缀重命名

    import java.io.File; public class BatchFileSuffixRename { public static void main(String[] args) { / ...