P1537 弹珠

题目描述

玛莎和比尔各自有自己的弹珠收藏。他们想重新分配收藏品,使两人能平等拥有弹珠。如果所有的弹珠的价值相同,那么他们就可以平分。但不幸的是,有一些弹珠更大,或者更美丽,所以,玛莎和比尔给每个弹珠一个1到6的价值。现在他们想平分这些弹珠,使每个人得到的总价值相同。不幸的是,他们发现,他们可能无法以这种方式分弹珠(即使弹珠的总价值为偶数)。例如,如果有一个价值为1、一个价值为3和两个价值为4的弹珠,这样他们就不能把弹珠分为价值相等的两部分。因此,他们想要你写一个程序,告诉他们是否能将所有弹珠分成价值相等的两部分。

输入格式:

输入文件有若干行,行中包含六个非负整数N1,。..,N6,其中mi是数值i的弹珠的价值。最大弹珠总数将达到20000。

输入文件的最后一行是0 0 0 0 0 0 。不要处理这一行。

输出格式:

对于每一组数据,输出"Collection #k:", k为输出的是第几组, 接着是"Can be divided." 或 "Can't be divided.".

每一组输出后多打一个空行。

输入样例:

1 0 1 2 0 0

1 0 0 0 1 1

0 0 0 0 0 0

输出样例:

Collection #1:\ Can't be divided.

Collection #2:\ Can be divided.

思路:

仔细一想,很简单的背包。 bool f[i]为能否凑到i; f[sum/2]即为答案。

代码:

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

const int N=10;
const int M=130000;

int a[N];
int flag;
int n,sum;
bool f[M];

int main()  {
    n=1;
    while(1)  {
        flag=0;
        for(int i=1; i<=6; i++)  {
            scanf("%d",&a[i]);
            if(a[i]==0)
                flag++;
            sum+=a[i]*i;
        }
        if(flag==6)
            break;
        if(sum%2==1)
            printf("Collection #%d:\nCan't be divided.\n\n",n);
        else  {
            memset(f,0,sizeof(f));
            f[0]=1;
            for(int i=1; i<=6; i++)
                for(int j=sum/2; j>=i; j--)
                    for(int k=1; k<=a[i]; k++)
                        if(j>=i*k && f[j-i*k])
                            f[j]=1;
            if(f[sum/2])
                printf("Collection #%d:\nCan be divided.\n\n",n);
            else
                printf("Collection #%d:\nCan't be divided.\n\n",n);
        }
        sum=0;
        n++;
    }
    return 0;
}

 

posted @ 2019-08-02 21:42  双子最可爱啦  阅读(187)  评论(0编辑  收藏  举报