cf916b 思维

#include <iostream>
#include <cstdio>
#include <cstring>
#include <string>
#include <cmath>
using namespace std;

long long n;
long long k;

long long ct[200005];

long long fac[25];
int main() {
    fac[0]=1;
    for(int i=1;i<=22;i++)
    {
        fac[i]=fac[i-1]*2;
    }

    while(~scanf("%lld%lld",&n,&k))
    {
        memset(ct,0,sizeof(ct));

        long long tp=n;
        int sta=100000;
        long long cnt=0;
        while(tp>0)
        {
            int id=tp%2;
            ct[sta]+=id;
            cnt+=id;
            sta++;
            tp/=2;
        }

        if(k<cnt)
            printf("No\n");
        else
        {
            printf("Yes\n");
            sta--;
            k-=cnt;
            for(int i=sta;i>=0;i--)
            {
                long long cur=min(k,ct[i]);
                ct[i]-=cur;
                ct[i-1]+=cur*2;
                k-=cur;
                if(k==0)
                    break;
            }

            for(int i=200000;i>=0;i--)
                if(ct[i]!=0)
                {
                    sta=i;break;
                }

            int sed=ct[sta-1]/2;
            ct[sta]+=sed;
            ct[sta-1]-=sed*2;
            k+=sed;

            int end;
            for(int i=0;i<=200000;i++){
                if(ct[i]!=0)
                {
                    end=i;break;
                }
            }

            //cout<<end<<endl;
            for(int i=end;i>=0;i--)
            {
                if(k==0)
                    break;
                if(i>=20)
                {
                    ct[i]--;
                    ct[i-1]+=2;
                    k--;
                }
                else
                {
                    long long cap=fac[i]-1;
                    //cout<<cap<<endl;
                    long long tp=(k/cap);
                    if(tp*cap!=k)
                        tp++;


                    ct[i]-=tp;
                    ct[i-1]+=tp*2;
                    k-=tp;
                }

                if(k==0)
                    break;
            }


            for(int i=200000;i>=0;i--)
            {
                for(long long j=0;j<ct[i];j++)
                {
                    printf("%d ",i-100000);
                }
            }
            printf("\n");
        }
    }

    return 0;
}

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值