Time Limit: 1 second

Memory Limit: 128 MB

【问题描述】

Mini进入洞口前自然要先来到镇里的装备店买些装备。买每件装备都需要付出一定的体力点,同时也会获得一定的能力点(每种装备

都可以买无限次^_^,但人的体力点可是有限的哦^_^)。当同种装备挑选的数量>=5时,除了挑选这些物品所获得的能力点之外

还会获得额外的Q*K(K表示挑选的数量)点的能力点。但是,同样的,Mini也会付出更多的体力点,即体力点还将多付出Q*K(K

表示挑选的数量)点。挑选的物品越多,产生的力量就越大,体力点也会付出越多。不过,这额外获得的能力点和付出的体

力点是可以不要的。为了能获得最大的能力点,同时也不至于使自己过分劳累(体力恰好为0不算过分劳累),Mini应该如何购

买装备呢?

【输入格式】

第一行,两个数N和M,N表示有多少种装备,0<=N<=340,每种装备可以买无限次。M表示Mini所拥有的体力点数,0<=M<=500。

以下N行,第I行三个数, WI,QI,Q。WI表示买第I个装备所付出的体力点,QI表示买第I个装备能获得的能力点,Q表示当第I个装备

的数量>=5时所产生的共鸣点。1<=WI<=9000,0<=QI<=9000,0<=Q<=9000。

【输出格式】

一个数,表示在保证Mini不会过分劳累的情况下所能买到的最大能力点

Sample Input

5 87
35 48 7
3 57 6
45 9 24
3 4 0
5 7 7

Sample Output

1653

【题解】

这题比较烦的地方是,多出来的额外能力点。但是其有说,如果不想要的话,可以不用花费额外的体力获得额外的技能点。也就是说。我可以买一个东西买了6个,但是不消耗额外体力获得5*q+6*q能力点的情况是允许的。

考虑到这种情况。我们先不要管那些额外技能点。

先按照给出的w[i],c[i]进行完全背包的操作。

会有两层for

for i 和for j

j是容量。

然后我们不能等这两层循环都结束之后再去处理额外能力点的问题。而应该在i层循环中一边处理不考虑额外能力点的完全背包问题一边在完全背包结束后考虑额外能力点的问题。

因为额外能力点也是一个完全背包。所以其更新方式也是顺序更新。

同时在不考虑额外能力点的完全背包中。要记录f[j]对应的num[j],即f[j]是在买了几个装备后获得的值。然后在单独考虑额外能力点的完全背包中。我们同样枚举j,只有在num[j] >=4的时候才尝试更新,即如果num[j] >=4 则看一下ju = j + (num[j]+1)*q[i] 是否大于m,如果不大于,则查看

f[j+(num[j]+1)*q[i]] 和 f[j] + (num[j]+1)*q[i]的大小关系。如果前者小于后者,则更新前者。

同时修改num[j+(num[j]+1)*q[i]]的值为num[j]+1;

因为是以j为更新起点,所以是num[j] >=4,因为再买一个装备就变成5了。

这段题解可以根据代码来理解。

【代码】
#include <cstdio>
#include <cstring>
int n,m,w[400],c[400],q[400],f[600] = {0},num[600] = {0};

int main()
{
//freopen("F:\\rush.txt","r",stdin);
scanf("%d%d",&n,&m);
for (int i = 1;i <= n;i++)
scanf("%d%d%d",&w[i],&c[i],&q[i]);
for (int i = 1;i <= n;i++)
{
<span style="white-space:pre">			</span>memset(num,sizeof(num),0);
for (int j = w[i];j <= m;j++) //先进行一次完全背包 不考虑多出来的额外技能点
if (f[j] < f[j-w[i]] + c[i])
{
f[j] = f[j-w[i]] + c[i];
num[j] = num[j-w[i]] + 1;
}
for (int j = 0;j <= m;j++) //考虑额外技能点,因为也是完全背包所以顺序枚举
if (num[j] >= 4) //如果数目大于等于4再更新一个装备就能>=5了
{
int ju = j+(num[j]+1) * q[i]; //这是所需要的花费
if (ju > m) continue;
if (f[ju] < f[j] + q[i] * (num[j]+1))
{
f[ju] = f[j] + q[i] * (num[j]+1); //加上相应的价值 看看是否更优
num[ju] = num[j] + 1; //同时要更新数目
}
}
}
printf("%d",f[m]);
return 0;
}

【t067】补充装备的更多相关文章

随机推荐

  1. android 闹钟提醒并且在锁屏下弹出Dialog对话框并播放铃声和震动

    android 闹钟提醒并且在锁屏下弹出Dialog对话框并播放铃声和震动            1.先简单设置一个闹钟提醒事件: //设置闹钟 mSetting.setOnClickListener ...

  2. es6常用功能与异步详解(JS高级面试题)

    callback hell方法的使用 可读性不友好 function loadImg(src,callback,fail){ var img = document.createElement('img ...

  3. C#集合类:动态数组、队列、栈、哈希表、字典

    1.动态数组:ArrayList 主要方法:Add.AddRange.RemoveAt.Remove 2.队列:Queue 主要方法:Enqueue入队列.Dequeue出队列.Peek返回Queue ...

  4. Day2:字符串常用方法

    字符串常用方法 #!/usr/bin/env python # -*- coding:utf-8 -*- # Author:Hiuhung Wan name = "my \tname is ...

  5. Android 仿今日头条频道管理(下)(GridView之间Item的移动和拖拽)

    前言 上篇博客我们说到了今日头条频道管理的操作交互体验,我也介绍了2个GridView之间Item的相互移动.详情请參考:Android 仿今日头条频道管理(上)(GridView之间Item的移动和 ...

  6. 模拟登录QQ推断是否须要验证码

    老生常谈的问题了,在模拟登录之前,推断是否须要验证码: https://ssl.ptlogin2.qq.com/check? uin=QQ号码&appid=1003903&js_ver ...

  7. POJ 1932 XYZZY (ZOJ 1935)SPFA+floyd

    http://poj.org/problem?id=1932 http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=1935 题目大 ...

  8. 浅谈求lca

    lca即最近公共祖先,求最近公共祖先的方法大概有3种,其实是窝只听说过3种,这3种做法分别是倍增求lca,树剖求lca和tarjan求lca,但是窝只会前2种,所以这里只说前2种算法了. 首先是倍增求 ...

  9. Net知识

    Net知识图谱   对于Web系统开发来说,Net其实也是有好多知识点需要学的,虽然目前JAVA是主流,就业市场比较大,但Net也在积极的拥抱开源,大Net Core 2 出来了,这无疑给Net开发者 ...

  10. 24、驱动调试之printk

    1.uboot跳转到内核启动的时候通过环境变量 console设置控制台 (console = ttySAC0表示输出到串口,并从串口结束输入,也可以设置console=tty1,表示输出到LCD,从 ...