可以对所有数建一个01字典树,然后可以发现每个点找最小的异或值就是在字典树中离它最近的点(可以把整个二叉树都画出来会更好看一点),对于每一个非叶子结点它的两个子树中的点如果想成为连通图的话,只能是两个子节点中数量都不能超过两个,不然就是两个连通图了,所以保留两个中的最大值另一个删得只剩一个值就好。对所有的非叶子结点都做这样的操作后就是最优结果了,将空节点减掉时间复杂度就能接受了。可以发现每个节点的子树会表示一个数的范围,所以不用建字典树就可以。
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int mod=1e9+7;
const int N=5e5+7,M=1e6+7;
int t,n,m;
ll a[N],b[N];
map<int,int> mp;
int find(ll l,ll r){
l=lower_bound(a+1,a+n+1,l)-a;
r=upper_bound(a+1,a+n+1,r)-a-1;
return r-l+1;
}
int divide(ll l,ll r){
if(find(l,r)==0||l>r) return 0;
if(l==r) return mp[l];
ll res=(r-l+1)>>1;
int res1=divide(l,l+res-1);
int res2=divide(l+res,r);
return max(res1,res2)+(min(res1,res2)>0);
}
void solve(){
scanf("%d",&n);
for(int i=1;i<=n;++i) scanf("%d",a+i),mp[a[i]]++;
sort(a+1,a+n+1);
printf("%d\n",n-divide(0,(1ll<<31)-1));
}
int main(){
t=1;
while(t--){
solve();
}
return 0;
}