CF 1446 C Xor Tree (分治)

利用01字典树解决CF 1446 C题,通过分治策略找到每个节点的最小异或值。非叶子节点的子树若要保持连通,其两个子节点中最多各保留两个不同值,否则会产生两个独立连通图。通过删除多余的值,得到最优解,最终实现O(n)的时间复杂度。
摘要由CSDN通过智能技术生成

在这里插入图片描述
在这里插入图片描述
可以对所有数建一个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;
}


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值