描述
有n个正整数,找出其中和为t(t也是正整数)的可能的组合方式。如:n=5,
5个数分别为1,2,3,4,5,t=5;
那么可能的组合有5=1+4和5=2+3和5=5三种组合方式。输入输入的第一行是两个正整数n和t,用空格隔开,其中1<=n<=20,表示正整数的个数,t为要求的和(1<=t<=1000)
接下来的一行是n个正整数,用空格隔开。输出和为t的不同的组合方式的数目。
样例输入
5 5 1 2 3 4 5
样例输出
3
方法一:
数据范围那么水,可以dfs
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
int data[30];
int main()
{
int n, t;
int ans = 0;
scanf("%d %d", &n, &t);
for(int i = 0; i < n; i++)
scanf("%d", data+i);
for(int i = 0; i < (1<<n); i++)
{
int sum = 0;
for(int j = 0; j < n; j++)
if(i&(1<<j))
{
sum += data[j];
}
if(sum == t)
ans++;
}
printf("%d\n", ans);
return 0;
}
方法二:
背包DP
dp[i]表示和为i时的最大方案数
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
int num[23333], dp[23333];
int main()
{
int n, t;
cin>>n>>t;
dp[0] = 1;
for(int i = 1; i<=n;i++)
cin>>num[i];
for(int i=1;i <= n;i ++)
for(int j = t; j >= num[i]; j--)
dp[j] += dp[j-num[i]];
cout<<dp[t];
return 0;
}