subset

题目:

将1,2,3,4,5……n-1,n分成和相等的两个子集,有多少种分法

#include<stdio.h>
#include<math.h>
#include<assert.h>

int main()
{
FILE *fin,*fout;
fin=fopen("1.txt","r");
fout=fopen("2.txt","w");
assert(fin);
assert(fout);

int N,dp[40][800]={0};
fscanf(fin,"%d",&N);
dp[1][1]=1;
int i,j,max=1,maxt=1;
for(i=2;i<=N;i++)
{
for(j=0;j<=max;j++)
{
dp[i][j+i]+=dp[i-1][j];
dp[i][abs(j-i)]+=dp[i-1][j];
if((j+i)>maxt) maxt=j+i;
}
max=maxt;
for(j=0;j<max;j++)
fprintf(fout,"%d ",dp[i][j]);
fprintf(fout,"\n");
}
fprintf(fout,"%d\n",dp[N][0]);
fclose(fin);
fclose(fout);
return 0;
}
1行01000000000000
2行01010000000000
3行10101000000000
4行10202010101000

 dp[i][j]的值表示1,2……i这几个值中的两个子集的和的差为j的个数

 例如dp[1][1]表示数字1形成两个集合,两个集合只能是一个为空集,一个仅含元素1.所以初始时dp[1][1]=1;

 前(i-1)个数组成的双子集差为j的个数为n; 前i个数组成的双子集差可以看成是向前(i-1)个数种添加一个i ,任意一个双子集都有两种添法,组成的新双子集j+i和|j-i|两种   况。

  从2~N循环,每次更新第i行的数值,最后输出mat[N][0]就可以得到结果。

    

 

 

 



转载于:https://www.cnblogs.com/zxs2011/archive/2012/03/25/2416925.html

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值