饭卡消费java_hdu 2546 饭卡(背包)

设饭卡余额为total

此题经分析 可以得出:要求选出一些饭菜 时消费量尽量接近total-5元 然后再买一个饭菜 以达到透支。。。

可以证明 最后买的那个饭菜是饭菜中价值最大的.

证明

设a1 a2 a3...an-1 an 为各饭菜的价格  设an的价格最大

sum=total-5

a1+a2+a3+...an-2+an-1+an=M

a1+a2+a3+...+an-2+an-1=x1          最后加an (按5元为界限)此时超额(an-1)-(sum-x1)=an-sum+a1+a2+...+an-2+an-1元                1

a1+a2+a3+...+an-2+an=x2             最后加an-1(按5元为界限)此时超额(an)-(sum-x2)=(an-1)-sum+a1+a2+...+an-2+an元              2

1式-2式=2*((an)-(an-1))>0

所以1式超额更多

所以最后选价格最大的那个饭菜得证

此题注意一点: 如果余额total本身就少于5元  直接输出total     此时没法买东西了  这种特殊情况要判断

动态规划

01背包问题 背包总量为余额-5  每个背包价值和重量都为w

#include

#include

#include

#define max(a,b) a>b?a:b

int cmp(const void *a,const void *b)

{

return *(int*)a-*(int*)b;

}

int main()

{

int n,a[5000],i,bb[5000],m,j;

while(scanf("%d",&n),n!=0)

{

memset(bb,0,sizeof(bb));

for(i=0;i

scanf("%d",&a[i]);

scanf("%d",&m);

qsort(a,n,sizeof(a[0]),cmp);

if(m<5)

{

printf("%d\n",m);

continue;

}

memset(bb,0,sizeof(bb));

m=m-5;

bb[0]=0;

for(i=0;i

{

for(j=m;j>=a[i];j--)

{

bb[j]=max(bb[j],bb[j-a[i]]+a[i]);

}

}

printf("%d\n",m+5-bb[m]-a[n-1]);

}

return 0;

}

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2546

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值