CodeVS1025 选菜 解题报告【背包型DP】

来源:互联网 发布:js获取当前系统时间 编辑:程序博客网 时间:2024/06/15 06:34

题目描述 Description
在小松宿舍楼下的不远处,有PK大学最不错的一个食堂——The Farmer’s Canteen(NM食堂)。由于该食堂的菜都很不错,价格也公道,所以很多人都喜欢来这边吃饭。The Farmer’s Canteen的点菜方式如同在超市自选商品一样,人们从一个指定的路口进去,再从一个指定的路口出来并付款。由于来这里就餐的人数比较多,所以人们自觉地在进入口的时候就排成一个长队,沿着长长的摆放着各式各样佳肴的桌子进行选菜。
小松发现,这种选菜方式意味着,他不能在选菜的时候离开队伍去拿一些他已经看过了的菜或者没有看过的菜,因为插队是不礼貌的,也是被BS的。
每个菜有一个价值,而小松也自己给每个菜定了一个在他看来的美味价值,例如红烧小黄鱼在小松看来是美味价值很高的,而花菜在小松眼里则是美味价值极低的菜肴。而有一些菜是营养价值极其高的菜(例如米饭),所以无论它的美味价值是多少,小松都会选择1份。现在小松带了X元钱来食堂就餐,他想知道,在不欠帐的情况下,他选菜的美味价值总合最大是多少。
输入描述 Input Description
请从输入文件farmer.in中读入相关数据。输入的第一行包括两个个整数n(1≤n≤100),k(0≤k≤实际菜的种类)和一个实数X(0≤X≤100),表示有n个菜式,有k种菜是必选的,小松带来了X元钱(精确到“角”)。接下来的1行包含n个实数,表示菜桌上从入口到出口的所有菜的价格(0≤价格≤10,单位“元”,精确到“角”);再接下来的1行包含n个整数,表示菜桌上从入口到出口的所有菜的美味价值(0≤美味价值≤100);再接下来一行包含n个整数,表示菜桌上从入口到出口的所有菜的种类编号(1≤种类编号≤100)。最后一行包含k个整数,分别表示必选菜的种类编号。要注意的是,同一种编号的菜可以出现多次,但是他们的价格和美味价值都是一样的。对于同一种菜(无论是不是必选菜),小松最多只会选择1份(买两份红烧豆腐多没意思啊)。另外,必选菜的价格之和一定不超过X。
输出描述 Output Description
请将结果输出到输出文件farmer.out中。输出包含一个整数,表示小松能选到的菜的美味价值总和最大是多少。
注:你可以假设数据中不会出现小松带的钱不够买必买菜的情况。
样例输入 Sample Input
7 1 5.0
4 1 3 0.9 2 0.5 0.9
7 3 5 2 5 0 2
6 3 5 2 4 1 2
2
样例输出 Sample Output
10
解题报告
题意就是有一些菜,每个菜有价格和美味程度,有一些菜是必选的,我们的钱有限,求最大的美味程度。
显然是一个典型的背包问题,就是要考虑必选的不能重选。

#include<cstdio>#include<iostream>#include<algorithm>using namespace std;const int N=100;int n,k,m,mon,s;int v[N+5],f[N*N+5];bool b[N+5];double p[N+5];double x;struct node{    int p,v;}a[N+5];int main(){    scanf("%d%d%lf",&n,&k,&x);    for(int i=1;i<=n;i++)scanf("%lf",&p[i]);    for(int i=1;i<=n;i++)scanf("%d",&v[i]);    for(int i=1;i<=n;i++)    {        scanf("%d",&m);        a[m].p=p[i]*10;//用int处理,谢绝卡精度         a[m].v=v[i];b[m]=true;//打好标记,方便以后的DP     }    mon=x*10;    for(int i=1;i<=k;i++)    {        scanf("%d",&m);        if(b[m]){mon-=a[m].p;s+=a[m].v;b[m]=false;}    }    for(int i=1;i<=n;i++)    if(b[i])    for(int j=mon;j>=a[i].p;j--)    f[j]=max(f[j],f[j-a[i].p]+a[i].v);    int ss=0;    for(int i=0;i<=mon;i++)ss=max(ss,f[i]);    printf("%d",s+ss);}
原创粉丝点击