对于由从1到N (1 <= N <= 39)这N个连续的整数组成的集合来说,我们有时可以将集合分成两个部分和相同的子集合。
例如,N=3时,可以将集合{1, 2, 3} 分为{1,2}和{3}。此时称有一种方式(即与顺序无关)。
N=7时,共有四种方式可以将集合{1, 2, 3, ..., 7} 分为两个部分和相同的子集合:
{1,6,7} 和 {2,3,4,5}
{2,5,7} 和 {1,3,4,6}
{3,4,7} 和 {1,2,5,6}
{1,2,4,7} 和 {3,5,6}
输入:程序从标准输入读入数据,只有一组测试用例。如上所述的N。
输出:方式数。若不存在这样的拆分,则输出0。
测试输入 | 期待的输出 | 时间限制 | 内存限制 | 额外进程 | |
---|---|---|---|---|---|
测试用例 1 | 以文本方式显示
| 以文本方式显示
| 1秒 | 64M | 0 |
#include<iostream>
#include<cstdio>
using namespace std;
int dp[1000]={1}; //第一个值初始化为1
int main(){
int n;
scanf("%d",&n);
if(n*(n+1)%4)
{
printf("0\n");
return 0;
}
int hs=n*(n+1)/4; //总和的一半
for(int i=1;i<=n;++i)
{
for(int j=hs;j>=i;--j) //更新组成和为j的组数
{
dp[j]+=dp[j-i];
//cout<<j<<" "<<dp[j]<<endl;
}
}
printf("%d\n",dp[hs]/2); //去掉重复的一半
return 0;
}