第n遍01背包的问题,还是没做出来,还是个难度系数为2的题目唉。。。
找数达人
时间限制:
1000 ms | 内存限制:
65535 KB
难度:2
-
描述
-
简单的题意如下:
小明最近做出了一道题:如何在一组数中寻找三个数,这三个数的和等于一给出的定值m,洋洋得意。于是小华不乐意了,别问为什么...,于是小华说,你能找找在一组数字中是否有n个数,使得这n个数的和等于一给定的定值m吗?
1): 0 < n <= 10000
2): 0 < 序列长度 <= 10000
3): 0 <= m <=10000
4): 0 <= 数组中的数 <=10000
还有比这更简单易懂的题目吗???
-
输入
- 多组数据,每组两行,第一行为序列内容,第二行为m的值。 输出
- 如能找到,输出Yes,否则,输出No。 样例输入
-
1 3 4 5
-
5
-
1 3 4 5 9
-
2
样例输出
-
Yes
-
No
这个题目是01背包的变形,刚开始想到了二叉树的解法,很显然不大好实现,或者会超时,那么01背包如何实现呢
其实将题目的现状摆出,与01背包问题相对应:
也就是说:01背包问题:
在最大重量为m的情况下在n件物品中寻找最大价值val:dp[n][m]=max(dp[n-1][m],dp[n-1][n-w[i]]+v[i]);
现在是在无限制(也算是最大重量为val吧)的情况下,寻找在n个数字中寻找价值val:dp[n][m]=max(dp[n-1][m],dp[n-1][m-val[i]]+val[i]);
也就是这么对应了,就可以求出相对应的转移公式!!!
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
#include <cmath>
using namespace std;
int main()
{
char str[30000];
int num[10005];
int dp[10005];
while(gets(str))
{
memset(dp,0,sizeof(dp));
memset(num,0,sizeof(num));
int i=0,j=0;
int flag=0;
int m;
int len=strlen(str);
while(i<len)
{
int sum=0;
while(str[i]>='0'&&str[i]<='9')
{
sum=sum*10+str[i]-'0';
i++;
}
num[j++]=sum;
i++;
}
scanf("%d",&m);
for(int i=0;i<j;i++)
{
for(int k=m;k>=num[i];k--)
{
dp[k]=max(dp[k],dp[k-num[i]]+num[i]);
if(dp[k]==m)
{
flag=1;
break;
}
}
if(flag)
{
break;
}
}
if(flag)
{
printf("Yes\n");
}
else
{
printf("No\n");
}
getchar();
}
return 0;
}