luogu2431 正妹吃月饼

Posted headboy2002

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了luogu2431 正妹吃月饼相关的知识,希望对你有一定的参考价值。

题目大意

求一个正整数集合\(K\),使得\(\sum_{k\in K}2^k\in[A,B]\),且\(|K|\)最大。\(A,B\)大小在long long范围内。

思路

\(\sum_{k\in K}2^k\)?这不就是一个二进制数,对于该数上的每一个数位\(k\),若\(k\in K\),则该数位上的数为1,否则为0么?
所以原题就变成了:求一个整数\(x\),使得\(x\in[A,B]\),且其用二进制表示的1的个数最多。
怎么求这个\(x\)呢?

  1. \(A,B\)二进制最高位数不同,则结果为\(A\)最高位数-1。如\(A=(10100)_2,B=(101)_2\),则结果为4,因为\((1111)_2\)
  2. 满足情况1时的特殊情况。若\(A\)用二进制表示全是1,则结果为情况1的结果加1.如\(A=(11111)_2,B=(101)_2\),则结果为5,因为\((11111)_2\)
  3. 若最高位数相同,则把两个数的最高位去掉,转化为子问题,再在此基础上+1.如:\(A=(110100)_2,B=(100101)_2\),则结果为5,因为把两个数最高位的1去掉就成了情况1的例子,再在此基础上+1便是。

注意

  • 因为A、B是long long,所以不能直接用1进行左移操作(因为默认的结果是int)。
  • 递归时,当A、B都是0时要提前跳出。
#include <cstdio>
#include <cstring>
#include <iostream>
#include <cassert>
using namespace std;

#define ll long long

int GetHighbitPos(ll x)
{
    int ans = -1;
    while (x)
    {
        ans++;
        x >>= 1;
    }
    return ans;
}

ll All(int n)
{
    ll x = 1;
    return (x << n + 1) - 1;
}

ll Erase(ll x, int pos)
{
    ll t = 1;
    return x & ~(t << pos);
}

int Dfs(ll below, ll above)
{
    if (above == 0)
        return 0;
    int pa = GetHighbitPos(above), pb = GetHighbitPos(below), ans;
    assert(pa >= pb);
    if (pa > pb)
    {
        ans = pa;
        if (above == All(pa))
            ans++;
        return ans;
    }
    else
        return 1 + Dfs(Erase(below, pb), Erase(above, pa));
}

int main()
{
    ll below, above;
    cin >> below >> above;
    cout << Dfs(below, above) << endl;
}

以上是关于luogu2431 正妹吃月饼的主要内容,如果未能解决你的问题,请参考以下文章

正妹吃月饼

CC2431 代码分析-CC2431狂轰滥炸

CC2431 代码分析-CC2431 喊出第一声

阿里云抢月饼代码实验,自己写着玩的

月饼事件代码

用代码过中秋,python海龟月饼你要不要尝一口?