杭电 2546 饭卡

Problem Description

电子科大本部食堂的饭卡有一种很诡异的设计,即在购买之前判断余额。如果购买一个商品之前,卡上的剩余金额大于或等于5元,就一定可以购买成功(即使购买后卡上余额为负),否则无法购买(即使金额足够)。所以大家都希望尽量使卡上的余额最少。
某天,食堂中有n种菜出售,每种菜可购买一次。已知每种菜的价格以及卡上的余额,问最少可使卡上的余额为多少。

Input

多组数据。对于每组数据:
第一行为正整数n,表示菜的数量。n<=1000。
第二行包括n个正整数,表示每种菜的价格。价格不超过50。
第三行包括一个正整数m,表示卡上的余额。m<=1000。

n=0表示数据结束。

Output

对于每组输入,输出一行,包含一个整数,表示卡上可能的最小余额。

Sample Input

1
50

5

10

1 2 3 2 1 1 2 3 2 1

50

0

Sample Output

-45

32

背包问题,先求出饭卡买的东西的价值,再用饭卡的钱减去价值,注意先判断钱是不是大于5,先把东西的价值排序,算出买前n-1的物品的价值,因为卡可以欠费,所以最后卡的余额可能小于最后一件东西的价值,所以先算出n-1件东西的价值。

 1 #include<cstdio>
 2 #include<algorithm>
 3 #include<string.h>
 4 using namespace std;
 5 int main()
 6 {
 7     int n,dp[5050],a[5050];
 8     while(scanf("%d",&n)&&n)
 9     {
10         int i,j;
11         for(i = 1 ; i <= n ; i++)
12         {
13             scanf("%d",&a[i]);
14         }
15         int m;
16         scanf("%d",&m);
17         sort(a+1,a+n+1);
18         memset(dp,0,sizeof(dp));
19         if(m < 5)
20             printf("%d\n",m);
21         else
22         {
23             m-=5;        //余额小于5不能买东西,这里先减去5
24             for(i = 1 ; i < n ; i++)        //注意i < n
25             {
26                 for(j = m ; j >= 0 ; j--)
27                 {
28                     if(j >= a[i])
29                     dp[j] = max(dp[j],dp[j-a[i]]+a[i]);        //求出n-1件东西的价值(k里余额为正)
30                 }
31             }
32             printf("%d\n",m+5-dp[m]-a[n]);
33         }
34     }
35 }
时间: 2024-10-05 18:41:46

杭电 2546 饭卡的相关文章

杭电 2546 饭卡(01背包)

http://acm.hdu.edu.cn/showproblem.php?pid=2546 饭卡 Time Limit: 5000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submission(s): 11285    Accepted Submission(s): 3880 Problem Description 电子科大本部食堂的饭卡有一种很诡异的设计,即在购买之前判断余额.如果购买一

杭电 2546 饭卡【01背包】

解题思路:先忽略饭卡余额大于等于5块才能买饭这一细节,需要求的是饭卡里面剩余的钱最少,转化一下,变成花的钱最多,那么剩下的钱就最少,再考虑余额大于等于5块才能买饭这一细节,可以这样想,如果卡里的余额不足5块,那么买不到饭,直接输出现在卡里的金额,如果卡里的钱多于5块,我们就可以先将这5块钱留起来,这样保证它每一次买饭卡里的余额都是大于5块的,最后卡里剩下的5块钱则用来买最贵的菜,这样就需要对菜的价钱进行排序.经过这样的转化后就可以转化成一个容量为m-5的包怎样装获得最大价值的01背包问题了. 反

hdu 2546 饭卡

0-1背包问题 i = 1,扫所有上限价钱只购买一件物品的最大消费(price[1]) i = 2,更新一遍,此时是购买两件物品的最大消费(price[2]) 以此类推~有n件物品 但是只进行到n-1,是因为最大的那件物品留至最后才减 价值最高上限为m-5 1 #include<iostream> 2 #include<memory.h> 3 #include<algorithm> 4 using namespace std; 5 int price[1010]; 6

hdu 2546 饭卡(0-1背包)

题目来源:hdu 2546 饭卡 饭卡 Time Limit: 5000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total Submission(s): 16645 Accepted Submission(s): 5797 Problem Description 电子科大本部食堂的饭卡有一种很诡异的设计,即在购买之前判断余额.如果购买一个商品之前,卡上的剩余金额大于或等于5元,就一定可以购买成功(即使购买后

HDU 2546 饭卡 01背包问题

01背包问题,增加一个额外条件,余额大于等于5的时候,可以购买任意价格的东西,那么就把5保留下来买最大价格的物品. 然后转化为收益的时候要注意初始条件,即没买东西的时候收益就是余额,就是所有的钱.这个还是有点难转换思维的. 最后是要熟悉背包填表,不需要保留选择的时候,就逆向填表,只需要一维表就可以了,当然也可以使用滚动数组,两个一维数组就可以了. 走了算法一圈回来了,AC自动机,线段树等都学会了,但是遇上这些基础DP还是不太轻松,看来需要继续打打基础.主要是思维转换不够流畅,速度就不快了. #i

饭卡------HDOJ杭电2546(还是01背包!!!!!!)

Problem Description 电子科大本部食堂的饭卡有一种非常诡异的设计,即在购买之前推断剩余金额. 假设购买一个商品之前,卡上的剩余金额大于或等于5元,就一定能够购买成功(即使购买后卡上剩余金额为负),否则无法购买(即使金额足够).所以大家都希望尽量使卡上的剩余金额最少. 某天,食堂中有n种菜出售.每种菜可购买一次. 已知每种菜的价格以及卡上的剩余金额,问最少可使卡上的剩余金额为多少. Input 多组数据.对于每组数据: 第一行为正整数n.表示菜的数量.n<=1000. 第二行包含

HDU 2546 饭卡(0-1背包)

http://acm.hdu.edu.cn/showproblem.php?pid=2546 题意: 电子科大本部食堂的饭卡有一种很诡异的设计,即在购买之前判断余额.如果购买一个商品之前,卡上的剩余金额大于或等于5元,就一定可以购买成功(即使购买后卡上余额为负),否则无法购买(即使金额足够).所以大家都希望尽量使卡上的余额最少. 某天,食堂中有n种菜出售,每种菜可购买一次.已知每种菜的价格以及卡上的余额,问最少可使卡上的余额为多少. 思路:我们先用一个背包容量为price-5的背包去装菜,背包里

HDU 2546 饭卡(01 背包)

链接:http://acm.hdu.edu.cn/showproblem.php?pid=2546 思路:需要首先处理一下的的01背包,当饭卡余额大于等于5时,是什么都能买的,所以题目要饭卡余额最小,那预留5元(相当于饭卡余额为5)来买最贵的菜 然后对剩下n-1进行01背包dp才是正确的.但是还存在一个问题,那就饭卡初始余额小于5时,也要处理掉. 下面讲01背包(原型可以看大牛的背包九讲,本人也正在学习),定义dp[i][j]为买前i种菜品剩下j元时的最大消费值等于下面两中情况之一的值 有两种来

HDUOJ 2546 饭卡

饭卡 Time Limit: 1000ms Memory Limit: 32768KB This problem will be judged on HDU. Original ID: 254664-bit integer IO format: %I64d      Java class name: Main 电子科大本部食堂的饭卡有一种很诡异的设计,即在购买之前判断余额.如果购买一个商品之前,卡上的剩余金额大于或等于5元,就一定可以购买成功(即使购买后卡上余额为负),否则无法购买(即使金额足够