[ 9.22 ]CF每日一题系列—— 484A Bits

Description:
  给你一个l,r的区间让你找一个最小的x并且其二进制数要包含最多的1位,输出它的十进制

Solution:
  我本来就是贪心,但是贪大了,想1一直往上添加1,但是忘记了0在中间的情况,考虑好了之后,发现这样贪是错误的,因为越往后位数越大,所以你最后的结果只能是1,11,111,1111,11111,而不可能是10,100,1001,10010,所以应该从后往前贪,这也是我们十进制转二进制的人工算法吧算是,十进制转二进制,手算的时候,就是一步一步确定最高位的1在哪里,

code

  从满1开始用异或变0,如果变成了0小于l,这是不允许的,第一个满足条件的就是最好的,可能你有疑惑,不是要最小的x吗,你想想110,和101,区间假如就是[5,6]从111开始第一位不能变成0,第二位可以101就结束了,所以从后往前变零就是从最小满足到最大满足的考虑了

#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring>
using namespace std;
typedef long long int ll;

int main()
{
    ll num;
    ll l,r;
    int t;
    scanf("%d",&t);
    while(t--)
    {
        num = (1LL << 62) - 1;
        scanf("%lld%lld",&l,&r);
        for(int i = 61;i >= 0;i--)
        {
            //即时退出保证1是最多的
            if(num >= l && num <= r)
                break;
            if((num ^ (1LL << i)) >= l)
                num ^= (1LL << i);
        }
        printf("%lld\n",num);
    }
    return 0;
}

 

  

posted @ 2018-09-22 09:27  Butterflier  阅读(262)  评论(0编辑  收藏  举报