hdu_3466(01背包)

其实,就是让C商品的q不等于p,其他都相同,这时,你就会发现如果要买C商品的话,肯定得先买C商品,因为买C商品的代价最大。所以,我们可以按照qi-pi的顺序来确定大顺序。这里我们还可以用更严谨的方式来证明一下,比如A:p1 q1, B:p2 q2,然后,假设单独买A或者B的话,都是可以买到的。这时,若先买A,则你至少需要p1+q2的钱;若先买B,则至少需要p2+q1的钱。那肯定是花最少的钱咯,所以如果先买A再买B,那么p1+q2<p2+q1,转换一下,就是q1-p1>q2-p2,也就是说qi-pi大的先买。这里还得注意一点就是,排序的时候,得按照qi-pi从小到大排序,因为你买第n件商品的时候,是在比较你是否要先买第n件商品。打个比方让大家更好地理解,比如说f(3, 10),是不是max(f(2, 10-p3)+v3, f(2, 10)),你会发现这个第一种情况f(2,10-p3)+v3中,是先买了第三件商品,也就是说排在后面的商品会先买。好的,排好序之后,就把问题就转换为不需要考虑顺序的问题了,那就是解决0/1背包问题了。

 1 #include<cstdio>
 2 #include<cstring>
 3 #include<algorithm>
 4 using namespace std;
 5 struct item{
 6     int p;
 7     int q;
 8     int v;
 9 } I[550];
10 bool cmp(item a,item b){
11     return a.q-a.p < b.q-b.p;
12 }
13 int dp[5550];
14 int main()
15 {
16     int n,m;
17     while(~scanf("%d%d",&n,&m)){
18         memset(dp,0,sizeof(dp));
19         for(int i = 0; i < n; i++){
20             scanf("%d%d%d",&I[i].p,&I[i].q,&I[i].v);
21         }
22         sort(I,I+n,cmp);
23         for(int j = 0;j < n ;j++){
24             for(int i = m; i >= max(I[j].q,I[j].p); i--){
25                 dp[i] = max(dp[i],dp[i-I[j].p]+I[j].v);
26             }
27         }
28         printf("%d\n",dp[m]);
29     }
30 
31     return 0;
32 }

猜你喜欢

转载自www.cnblogs.com/shanyr/p/11823633.html