码迷,mamicode.com
首页 > 其他好文 > 详细

D. Fill The Bag

时间:2020-02-13 09:47:54      阅读:64      评论:0      收藏:0      [点我收藏+]

标签:string   ref   遍历   大小   二进制   lld   bag   scan   相等   

题目链接:http://codeforces.com/contest/1303/problem/D

题意:给你一个大小为m的包,和n个盒子,每个盒子大小为a[i],且a[i]是2的正整数次幂,每个盒子可以拆分成2个大小相等的盒子,问至少经过几次拆分,才能正好装满m的包,如               果不可能则输出-1。

思路:把m化成2进制存入b数组中,由于a[i]是2的正整数次幂,那用c数组统计2的对应次幂分别有多少个,然后遍历m的二进制从低位到高位,如果对应的c数组有,那就c[i]--,否则               的话就看c[1]到c[i-1]是否可以撮成,可以的话就要那些低位撮,不可以的话就找一个高位进行拆分,然后累加拆分次数即可。

#include<stdio.h>
#include<iostream>
#include<string.h>
#include<math.h>
#include<string>
#include<algorithm>
#include<queue>
#include<map>
typedef long long ll;
using namespace std;
ll lo(ll x)
{
    ll v=0;
    while(x)
    {
        v++;
        x/=2;
    }
    return v;
}
ll b[100],c[100];
int main()
{
    int t;
    scanf("%d",&t);
    while(t--)
    {
        memset(b,0,sizeof(b));
        memset(c,0,sizeof(c));
        ll m,n;
        scanf("%lld%lld",&m,&n);
        ll sum=0;
        for(int i=1;i<=n;i++)
        {
            ll x;
            scanf("%lld",&x);
            sum+=x;
            c[lo(x)]++;
        }
        if(sum<m)
        {
            printf("-1\n");
            continue;
        }
        int x=0;
        while(m)
        {
            x++;
            b[x]=m%2;
            m/=2;
        }
        ll ans=0;
        for(int i=1;i<=65;i++)
        {
            if(b[i]==0)
                continue;
            if(c[i]>0)
                c[i]--;
            else
            {
                ll y=2,z=-1;
                for(int j=i-1;j>=1;j--)//看是否可以撮成i,
                {
                    if(c[j]>=y)
                    {
                        c[j]-=y;
                        z=j+1;
                        break;
                    }
                    else
                        y-=c[j];
                    y*=2;
                }
                if(z!=-1)//可以则对应的减掉
                {
                    for(int j=z;j<i;j++)
                        c[j]=0;
                }
                else
                {
                    for(int j=i+1;j<=65;j++)//不可以则找高位拆分
                    {
                        if(c[j]>0)
                        {
                            c[j]--;
                            for(int k=j-1;k>=i;k--)
                            {
                                c[k]++;
                                ans++;
                            }
                            break;
                        }
                    }
                }
            }
        }
        printf("%lld\n",ans);
    }
}

 

D. Fill The Bag

标签:string   ref   遍历   大小   二进制   lld   bag   scan   相等   

原文地址:https://www.cnblogs.com/zcb123456789/p/12302062.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!