题意:有0,1,2三种任务,0任务中的任务至少得完成一件,1中的任务最多完成1件,2中的任务随便做。每一个任务最多只能做一次 。n代表有n组任务,t代表有t分钟,m代表这组任务有m个子任务,s代表这m个子任务属于0,1,2中的哪种类型,接下来是m个子任务,第一个数代表要花费的时间,第二个数代表得到的愉悦度......求在可以完成工作的情况的最大愉悦度....要是不能完成,输出-1(题意要求每个子任务只能被取一次)
错误思路:我一开始想,把0,1,2这三大组任务的子任务先统计好,在dp的时候,我开dp[3][105],代表在完成3组任务体积为105的情况的最大愉悦度.......这种思路是错的,因为题目给出的n组任务是有其固定顺序,只能是按照它给出来的解决问题.......
ac思路:分为n组,每一组判断这一组是属于0,1,2三种任务中的哪一组......
若是属于0,那么将这一组的dp[i][j],j从0~~t全部置为负无穷大,然后开始动态转移.....为什么要置为负无穷大?因为只有这样才能不出现一个都不选择的情况.....其动态转移方程,,在我前一个分组背包题目已经详细推导过,就是dp[i][j]=max(dp[i][j-v[i]]+val[i],dp[i-1][j-v[i]]+val[i],dp[i][j])
若是属于1,先将第i-1的状态传递到第i状态,在开始分组背包的模板......最多取一个dp[i][j]=max(dp[i-1][j],dp[i-1][j-v[i]]+val[i]);
若是属于2,先将第i-1的状态传递到第i状态,随意取,那么可以不取,取任意个子任务......dp[i][j]=max(dp[i-1][j],dp[i][j-v[i]]+val[i],dp[i-1][j-v[i]]+val[i]);
#include<iostream>
#include<stdio.h>
#include<string.h>
using namespace std;
#define M -1000000000
int dp[105][105],s[105][2];
int main()
{
int n,t;
while(scanf("%d %d",&n,&t)>0)
{
int m,k;
memset(dp,0,sizeof(dp));
for(int i=1;i<=n;i++)
{
scanf("%d %d",&m,&k);
for(int j=1;j<=m;j++)
scanf("%d %d",&s[j][0],&s[j][1]);
if(k!=0)
for(int j=0;j<=t;j++) // 传递状态
dp[i][j]=dp[i-1][j];
if(k==0) //至少取一个 ,分组背包变形模板.......
{
for(int tmp=0;tmp<=t;tmp++)
dp[i][tmp]=M;
for(int tmp=1;tmp<=m;tmp++) //这层for循环必须在前面......
{
for(int j=t;j>=0;j--)
{
if(j-s[tmp][0]>=0&&dp[i][j-s[tmp][0]]+s[tmp][1]>dp[i][j])
dp[i][j]=dp[i][j-s[tmp][0]]+s[tmp][1];
if(j-s[tmp][0]>=0&&dp[i-1][j-s[tmp][0]]+s[tmp][1]>dp[i][j])
dp[i][j]=dp[i-1][j-s[tmp][0]]+s[tmp][1];
}
}
}
else if(k==1) //最多取一个 ,分组背包模板
{
for(int j=t;j>=0;j--)
{
for(int tmp=1;tmp<=m;tmp++)
{
if(j-s[tmp][0]>=0)
{
if(dp[i-1][j-s[tmp][0]]+s[tmp][1]>dp[i][j])
dp[i][j]=dp[i-1][j-s[tmp][0]]+s[tmp][1];
}
}
}
}
else if(k==2) //随意取 ,01背包,每个子任务只能取一次,却可以取任意个不同的子任务
{
for(int tmp=1;tmp<=m;tmp++)
{
for(int j=t;j>=s[tmp][0];j--)
{
if(dp[i][j-s[tmp][0]]+s[tmp][1]>dp[i][j])
dp[i][j]=dp[i][j-s[tmp][0]]+s[tmp][1];
if(dp[i-1][j-s[tmp][0]]+s[tmp][1]>dp[i][j])
dp[i][j]=dp[i-1][j-s[tmp][0]]+s[tmp][1];
}
}
}
}
if(dp[n][t]<0)
printf("-1\n");
else
printf("%d\n",dp[n][t]);
}
return 0;
}