【HDU6687】Rikka with Stable Marriage(Trie树 贪心)

题目链接

大意

给定\(A,B\)两个数组,让他们进行匹配。
我们称\(A_i\)\(B_j\)的匹配是稳定的,当且仅当目前所剩元素不存在\(A_x\)\(B_y\)使得
\(A_i\oplus B_j<A_i\oplus B_y\)\(A_i\oplus B_j<A_x\oplus B_j\)成立
问所有稳定匹配的情况中\(A_i\oplus B_j\)之和最大的是多少。

思路

考虑找到当前异或和最大的一对\(A_i\oplus B_j\),那么不会存在一个\(A_x\)\(B_y\)可以使得它不稳定,所以这对\(A_i\oplus B_j\)一定会被计入答案。
所以我们每次只用找最大的\(A_i\oplus B_j\)就行了。

考虑如何查找出最大的\(A_i\oplus B_j\)

我们可以使用两颗Trie树分别维护出\(A,B\)的信息。
贪心地想,我们每次查找从高位开始,走完全相反的两个方向(如果可以走),一定会比不走相反的方向优。
这样查找可以满足双方面最大,即找出来的\(A_i,B_j\)一定是会计入答案的(尽管可能不是当前最大的\(A_i\oplus B_j\))。
因为如果该点对没计入到答案,那么当前一定会存在更优的某个\(A_x\)\(B_y\)使得它不稳定,那么在查找的途中,一定就会进入另一条更优的支路。
所以这样查找依旧可以满足答案的正确性。

代码

#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int MAXN=100005;
int K,N;
long long Ans;
struct Tree{
	int Cnt,Root;
	struct Node{
		int val,dep;
		int ch[2];
	}s[MAXN*31];
	void Insert(int rt,int val,int dep){
		s[rt].val++;s[rt].dep=dep;
		if(dep==-1)return ;
		int u=(1&(val>>dep));
		if(!s[rt].ch[u])s[rt].ch[u]=++Cnt;
		Insert(s[rt].ch[u],val,dep-1);
	}
}T1,T2;
void Clear(){
	for(int i=1;i<=T1.Cnt;i++)T1.s[i].ch[0]=T1.s[i].ch[1]=T1.s[i].dep=T1.s[i].val=0;
	for(int i=1;i<=T2.Cnt;i++)T2.s[i].ch[0]=T2.s[i].ch[1]=T2.s[i].dep=T2.s[i].val=0;
	T1.Root=T2.Root=T1.Cnt=T2.Cnt=1;Ans=0;
}
void Query(){
	int rt1=T1.Root,rt2=T2.Root;
	for(int i=30;i>=0;i--){
		if(T1.s[T1.s[rt1].ch[0]].val&&T2.s[T2.s[rt2].ch[1]].val)
			rt1=T1.s[rt1].ch[0],rt2=T2.s[rt2].ch[1],Ans+=(1<<i);
		else if(T1.s[T1.s[rt1].ch[1]].val&&T2.s[T2.s[rt2].ch[0]].val)
			rt1=T1.s[rt1].ch[1],rt2=T2.s[rt2].ch[0],Ans+=(1<<i);
		else if(T1.s[T1.s[rt1].ch[0]].val&&T2.s[T2.s[rt2].ch[0]].val)
			rt1=T1.s[rt1].ch[0],rt2=T2.s[rt2].ch[0];
		else if(T1.s[T1.s[rt1].ch[1]].val&&T2.s[T2.s[rt2].ch[1]].val)
			rt1=T1.s[rt1].ch[1],rt2=T2.s[rt2].ch[1];
		T1.s[rt1].val--;T2.s[rt2].val--;
	}
}
int main(){
	scanf("%d",&K);
	while(K--){
		Clear();
		scanf("%d",&N);
		for(int i=1,x;i<=N;i++)
			scanf("%d",&x),T1.Insert(T1.Root,x,30);
		for(int i=1,x;i<=N;i++)
			scanf("%d",&x),T2.Insert(T2.Root,x,30);
		for(int i=1;i<=N;i++)Query();
		printf("%lld\n",Ans);
	}
}

后记

其实题意不能简单地化为让\(A\)\(B\)两两匹配的异或和最大。
(因为要满足稳定条件,且若的按和最大去做会被以下数据卡掉)

1
2
1 2
2 7

正解为6,手跑和最大为8

posted @ 2019-11-11 21:59  孤攀客  阅读(233)  评论(0编辑  收藏  举报