2020牛客寒假算法基础集训营2 I题

建通道

题目描述

在无垠的宇宙中,有 n 个星球,第 i 个星球有权值 vi。

由于星球之间距离极远,因此想在有限的时间内在星际间旅行,就必须要在星球间建立传送通道。

任意两个星球之间均可以建立传送通道,不过花费并不一样。第 i 个星球与第 j 个星球的之间建立传送通道的花费是 lowbit(vi⊕vj),

其中 ⊕ 为二进制异或,而 lowbit(x) 为 x 二进制最低位 1 对应的值。

牛牛想在这 n 个星球间穿梭,于是――你需要告诉 牛牛,要使这 n 个星球相互可达,需要的花费最少是多少。


题目的思路就是找到一个最小的二进制位置,这个位置的二进制位值是1,并且这个位置在n个星球权值的二进制位值里面也存在为1,并且存在为0;

神奇的异或运算就解决了;

#include<bits/stdc++.h>
#define ll long long
#define pa pair<int,int>
#define lson k<<1
#define rson k<<1|1
//ios::sync_with_stdio(false);
#define inf 0x3f3f3f3f
using namespace std;
const int N=200100;
const int M=1000100;
const ll mod=100000007;
int a[N];
set<int>se;
ll ans;
int main(){
    ios::sync_with_stdio(false);
    int n;
    cin>>n;
    for(int i=1;i<=n;i++){
        cin>>a[i];
        se.insert(a[i]);
    }
    int s1=0,s2=0x7fffffff;
    for(int i=1;i<=n;i++){
        s1|=a[i];//找出二进制位有1的位置
        s2&=a[i];//找出二进制位都为1的位置
    }
    s2^=s1;//找出二进制位有1但不都为1的位置
    int t=se.size()-1;
    for(int i=0;i<=30;i++){
        int c=1<<i;
        if(s2&c){//找出s1二进制位的为1的最小位置
            ans=1ll*c*t;
            break;
        }
    }
	cout<<ans<<endl;
    return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值