今天拉了一场CF,做了一下,略坑啊、、、首先105A题,竟然卡精度,小数点两位卡精度,需要给他加一个1e-6,算是见识了
题意:给出一些n克面,以及m种馅儿,每种馅儿做面包需要的面的克数和馅儿的克数以及馅儿的总克数,面也可以单独做面包,然后每一种面包都有价格,求做的面包的总价格最高?
分析:很贱的题目啊,读了之后就开始贪心,贪心竟然过了10组数据,然后我以为有漏洞,后来发现是个多重背包。dp的题目最怕用贪心做了,幸亏发现了。
标准的多种背包,转化一下就行了,n克面就是背包容量,然后每种的数量是馅儿的总量 / 每个的,花费是面的克数,价值是价格。剩下的事情就是模板了、二进制优化一下
AC代码:
#include <cstdio> #include <cstring> #include <algorithm> #include <iostream> #include <map> #include <vector> using namespace std; const int inf = 0x3f3f3f3f; const int N = 1200; struct Node { int cost,w,nk; //价格,重量,代数 }; Node num[N]; int dp[N]; //int cmp(Node a,Node b) //{ // if(a.p!=b.p) // { // return a.p>b.p; // } //} void one(int cost , int w, int m) { for(int i=cost ; i<=m; i++) dp[i] = max(dp[i] , dp[i - cost ]+w); } void zero_one(int m ,int w, int cost) { for(int i=m; i>=cost; i--) dp[i]=max(dp[i],dp[i-cost]+w); } int main() { int n,m,a,b; scanf("%d%d%d%d",&n,&m,&a,&b); num[0].nk=n/a,num[0].cost=b,num[0].w=a; for(int i=1; i<=m; i++) { int x,y,z,h; scanf("%d%d%d%d",&x,&y,&z,&h); num[i].nk=x/y; num[i].cost=h,num[i].w=z; } for(int i=0; i<=m; i++) { int w=num[i].cost , cost=num[i].w, nk=num[i].nk; if(cost * nk >= n) one(cost , w , n); else { int k = 1; while( k < nk) { zero_one(n,k*w,k*cost); nk -= k; k *= 2; } zero_one(n,nk*w,nk*cost); } } printf("%d\n",dp[n]); return 0; }
原文地址:http://blog.csdn.net/y990041769/article/details/39555357