题意:

给你两个序列a,b,序列c的某位是由序列a,b的此位异或得来,让你重排序列ab,找出字典序最小的序列c。

题解:

如果能找到a,b序列中完全一样的值当然最好,要是找不到,那也尽量让低位不一样。

因此,将两个序列中元素的二进制视作字符串,建字典树。

在两棵字典树上,贪心地找让当前位一样的。

每找c的一位,dfs一遍,dfs过程中剪掉已经因为构造c而用掉的子树,构造序列c。

然后爆wa。。。

我造了好几组测例,发现贪心并不能找出当前位的最优解,比如dfs到两棵字典树上的某个点,这两个节点都既有左儿子,又有右儿子,那么到底是都向左走更优,还是都向右走更优,字典树上没有这个信息。

比如数据  3  1,0,2  2,4,1  程序打死都要给出我0,4,0的结果

直到有人提醒我:“你再把c数组排个序不就行了嘛”

我:“。。。。。。”

那么,不管不顾地贪心,只要dfs到的两点都能向1的方向走,那就都向1的方向走,因为当都向1这个方向走这棵子树此次被剪掉后,以后一定能构造出都向0的方向走的最优解。

只要再把c排序一遍,就不用计较此点最优解是先构造出来的还是后构造出来的了。

#include<iostream>
#include<cstring>
#include<algorithm>
using namespace std;
//const int MAXN=1<<30;
struct Node{
int to[];
bool end;
int size;
};
int c[];
struct Trie{
Node node[*];
int nodenum;
void init(int n){
for(int i=;i<=n*;i++)memset(&node[i],,sizeof (Node));
nodenum=;
}
void push(int n){
int nowid=;
for(int i=;i>=;i--){
++node[nowid].size;
bool t=(n>>i)&;
if(node[nowid].to[t]==){
node[nowid].to[t]=++nodenum;
nowid=nodenum;
}else{
nowid=node[nowid].to[t];
}
// n>>=1;
}
++node[nowid].size;
node[nowid].end=;
}
}trie1,trie2;
int find1(){
int id1=,id2=;
int ans=;
int i=;
bool to10,to11,to20,to21;
while(!trie1.node[id1].end && !trie2.node[id2].end){
ans<<=;
--trie1.node[id1].size;
--trie2.node[id2].size;
to10=trie1.node[trie1.node[id1].to[]].size>;
to11=trie1.node[trie1.node[id1].to[]].size>;
to20=trie2.node[trie2.node[id2].to[]].size>;
to21=trie2.node[trie2.node[id2].to[]].size>;
if(to11 && to21){
id1=trie1.node[id1].to[];
id2=trie2.node[id2].to[];
}else if(to10 && to20){
id1=trie1.node[id1].to[];
id2=trie2.node[id2].to[];
}else if(to10 && to21){
id1=trie1.node[id1].to[];
id2=trie2.node[id2].to[];
++ans;
}else if(to11 && to20){
id1=trie1.node[id1].to[];
id2=trie2.node[id2].to[];
++ans;
}
++i;
}
--trie1.node[id1].size;
--trie2.node[id2].size;
return ans;
}
int main(){
int t;
scanf("%d",&t);
while(t--){
int n,tmp;
scanf("%d",&n);
trie1.init(n);trie2.init(n);
for(int i=;i<=n;i++){
scanf("%d",&tmp);
trie1.push(tmp);
}
for(int i=;i<=n;i++){
scanf("%d",&tmp);
trie2.push(tmp);
}
// for(int i=1;i<=trie1.nodenum;i++){
// printf("%d %d %d %d\n",trie1.node[i].size,trie1.node[i].to[0],trie1.node[i].to[1],trie1.node[i].end);
// }
for(int i=;i<=n;i++){
c[i]=find1();
}
sort(c+,c++n);
for(int i=;i<=n;i++){
printf("%d",c[i]);
if(i<n)printf(" ");
else printf("\n");
}
}
}

hdu多校第五场1002 (hdu6625) three arrays 字典树/dfs的更多相关文章

  1. hdu多校第五场1005 (hdu6628) permutation 1 排列/康托展开/暴力

    题意: 定义一个排列的差分为后一项减前一项之差构成的数列,求对于n个数的排列,差分的字典序第k小的那个,n<=20,k<=1e4. 题解: 暴力打表找一遍规律,会发现,对于n个数的排列,如 ...

  2. 2014 HDU多校弟五场J题 【矩阵乘积】

    题意很简单,就是两个大矩阵相乘,然后求乘积. 用 Strassen算法 的话,当N的规模达到100左右就会StackOverFlow了 况且输入的数据范围可达到800,如果变量还不用全局变量的话连内存 ...

  3. 2014 HDU多校弟五场A题 【归并排序求逆序对】

    这题是2Y,第一次WA贡献给了没有long long 的答案QAQ 题意不难理解,解题方法不难. 先用归并排序求出原串中逆序对的个数然后拿来减去k即可,如果答案小于0,则取0 学习了归并排序求逆序对的 ...

  4. 2018杭电多校第五场1002(暴力DFS【数位】,剪枝)

    //never use translation#include<bits/stdc++.h>using namespace std;int k;char a[20];//储存每个数的数值i ...

  5. hdu多校第五场1004 (hdu6627) equation 1 计算几何

    题意: 给你一个C,再给你n组a,b,让你求x取什么值的时候,$ \sum_{i=1}^n |a_i*x+b_i| =C $,要求求出解的个数,并用最简分数从小到大表示,如果有无穷多解,输出-1. 题 ...

  6. hdu多校第五场1006 (hdu6629) string matching Ex-KMP

    题意: 给你一个暴力匹配字符串公共前缀后缀的程序,为你对于某个字符串,暴力匹配的次数是多少. 题解: 使用扩展kmp构造extend数组,在扩展kmp中,设原串S和模式串T. extend[i]表示T ...

  7. hdu多校第五场1007 (hdu6630) permutation 2 dp

    题意: 给你n个数,求如下限制条件下的排列数:1,第一位必须是x,2,最后一位必须是y,3,相邻两位之差小于等于2 题解: 如果x<y,那么考虑把整个数列翻转过来,减少讨论分支. 设dp[n]为 ...

  8. hdu多校第4场E. Matrix from Arrays HDU 二维前缀和

    Problem E. Matrix from Arrays Time Limit: / MS (Java/Others) Memory Limit: / K (Java/Others) Total S ...

  9. HDU多校第三场 Hdu6606 Distribution of books 线段树优化DP

    Hdu6606 Distribution of books 题意 把一段连续的数字分成k段,不能有空段且段和段之间不能有间隔,但是可以舍去一部分后缀数字,求\(min(max((\sum ai ))\ ...

随机推荐

  1. web框架express学习一

    服务端 node app.jsapp.jslet express = require("express"); let http = require("http" ...

  2. Nginx缓存配置指南

    1.如何配置基本缓存设置 开启简单的缓存配置,只需要两个指令:proxy_cache_path和proxy_cache.proxy_cache_path配置缓存的存放地址和其他的一些常用配置,prox ...

  3. 如何基于 Nacos 和 Sentinel ,实现灰度路由和流量防护一体化

    基于Alibaba Nacos和Sentinel,实现灰度路由和流量防护一体化的解决方案,发布在最新的 Nepxion Discovery 5.4.0 版,具体参考: 源码主页,请访问 源码主页指南主 ...

  4. SELinux导致PHP连接MySQL异常Can't connect to MySQL server的解决方法

    原文摘自:http://www.jb51.net/article/52581.htm 这篇文章主要介绍了SELinux导致PHP连接MySQL异常Can't connect to MySQL serv ...

  5. 基于mybatis-plus的代码生成

    基于mybatis-plus的代码生成 前言 随着敏捷开发模式的推广,伴着日益增长的需求,日常工作中我们越来越注重效率和便捷性.今天我们就来探讨下如何自动生成代码,准确地说是如何依赖数据库生成我们的e ...

  6. 关于Web前端密码加密是否有意义的总结

    关于Web前端密码加密是否有意义的总结    :    https://blog.csdn.net/hla199106/article/details/45114801 个人:加密涉及到的是前后端的数 ...

  7. 一道腾讯面试题:如何快速判断某 URL 是否在 20 亿的网址 URL 集合中?布隆过滤器

    何为布隆过滤器 还是以上面的例子为例: 判断逻辑: 多次哈希: Guava的BloomFilter 创建BloomFilter 最终还是调用: 使用: 算法特点 使用场景 假设遇到这样一个问题:一个网 ...

  8. Flink 编程模型

    抽象层次   levels_of_abstraction 最低级的抽象接口是状态化的数据流接口(stateful streaming).这个接口是通过 ProcessFunction 集成到 Data ...

  9. 020_JUC

    JUC Java.util.Concurrent 并发包 池的顶级接口 Executor 子接口 ExecutorService 工具类 Executors(Collections.Arrays .. ...

  10. 支付宝支付接口-运行支付宝demo

    运行deme 提供了 支付  查询 退款 交易关闭几个简单的接口demo 下载 https://docs.open.alipay.com/270/106291/ 转为mave项目 1.创建一个空的ma ...