
【nowcoder 214271】小D和他的魔法石
发布日期:2021-05-07 06:59:21
浏览次数:23
分类:精选文章
本文共 2946 字,大约阅读时间需要 9 分钟。
小D和他的魔法石
题目链接:
到牛客看:
题目大意
有一些魔法石,每种有无限个,用一个耗费一个值的体力,获得一个值的能力。
然后你可以有 k 次机会把两种魔法石的体力值互换,一定要把 k 次都用完。
然后问你在耗费体力不大于 m 的时候,最多能有多少能力值。
思路
这道题主要是考虑分类讨论。
首先,很明显的就是如果 k = 0 k=0 k=0,我们就直接做完全背包。
那考虑有换的机会,我们可以想到贪心。让一个魔法石耗费体力值最小,获得能力值最大。
那考虑怎么才能换出这个局面,怎么才换不出。
我们可以发现,k 就算是 1 1 1,也可以换出。但如果 n n n 是 2 2 2 时呢?
因为交换次数一定要用完,那我们就要判断换完之后是否是最优的。 是最优的就按这个输出,否则就又是背包。那对于 n n n 不是 2 2 2 的情况,那一定可以构造最优的石头。
代码
#include#include #include #include #include #define ll long longusing namespace std;int n, m, k, a[1001], b[1001], times;int main() { scanf("%d %d %d", &n, &m, &k); if (k == 0) { //不能变换,那就是一道普通的背包 for (int i = 1; i <= n; i++) scanf("%d", &a[i]); for (int i = 1; i <= n; i++) scanf("%d", &b[i]); ll f[2][1001], ans = 0; memset(f, 0, sizeof(f)); for (int i = 1; i <= n; i++) for (int j = m; j >= 0; j--) { f[i & 1][j] = f[(i - 1) & 1][j]; times = 1; while (j - a[i] * times >= 0) { f[i & 1][j] = max(f[i & 1][j], f[(i - 1) & 1][j - a[i] * times] + b[i] * times); times++; } } for (int i = 0; i <= m; i++) ans = max(ans, f[n & 1][i]); printf("%lld", ans); return 0; } if (n == 2) { //只有两种魔法石 int maxn = 0, minn = 2147483647; for (int i = 1; i <= n; i++) { scanf("%d", &a[i]); minn = min(minn, a[i]); } for (int i = 1; i <= n; i++) { scanf("%d", &b[i]); maxn = max(maxn, b[i]); } if ((a[1] == minn && b[1] != maxn) || (a[1] != minn && b[1] == maxn)) { //原本不是最优 if (k & 1) { //能在用完交换次数后达到我们要的最优状态 printf("%lld", 1ll * maxn * (1ll * (m / minn))); return 0; } else { //不能,那就是背包 ll f[2][1001], ans = 0; memset(f, 0, sizeof(f)); for (int i = 1; i <= n; i++) for (int j = m; j >= 0; j--) { f[i & 1][j] = f[(i - 1) & 1][j]; times = 1; while (j - a[i] * times >= 0) { f[i & 1][j] = max(f[i & 1][j], f[(i - 1) & 1][j - a[i] * times] + b[i] * times); times++; } } for (int i = 0; i <= m; i++) ans = max(ans, f[n & 1][i]); printf("%lld", ans); return 0; } } else { //原本是最优 if (k & 1) { //不能 swap(a[1], a[2]); ll f[2][1001], ans = 0; memset(f, 0, sizeof(f)); for (int i = 1; i <= n; i++) for (int j = m; j >= 0; j--) { f[i & 1][j] = f[(i - 1) & 1][j]; times = 1; while (j - a[i] * times >= 0) { f[i & 1][j] = max(f[i & 1][j], f[(i - 1) & 1][j - a[i] * times] + b[i] * times); times++; } } for (int i = 0; i <= m; i++) ans = max(ans, f[n & 1][i]); printf("%lld", ans); return 0; } else { //能 printf("%lld", 1ll * maxn * (1ll * (m / minn))); return 0; } } return 0; } int maxn = 0, minn = 2147483647; for (int i = 1; i <= n; i++) { scanf("%d", &a[i]); minn = min(minn, a[i]); } for (int i = 1; i <= n; i++) { scanf("%d", &b[i]); maxn = max(maxn, b[i]); } printf("%lld", 1ll * maxn * (1ll * (m / minn))); return 0;}
发表评论
最新留言
感谢大佬
[***.8.128.20]2025年03月26日 08时35分45秒
关于作者

喝酒易醉,品茶养心,人生如梦,品茶悟道,何以解忧?唯有杜康!
-- 愿君每日到此一游!
推荐文章
是什么?评估分类器的常用概念----准确率,精确率,召回率
2019-03-06
linux中使用awk命令
2019-03-06
LAB2 内核的内存管理
2019-03-06
如何使用google搜索?
2019-03-06
Redis分布式锁的正确实现方式
2019-03-06
设计模式-抽象工厂模式
2019-03-06
MySQL Explain查看执行计划详解
2019-03-06
IntelliJ IDEA 中,项目文件右键菜单没有svn选项解决办法
2019-03-06
Spring 动态绑定多实现类实例综述
2019-03-06
IDEA 调试Java代码的两个技巧
2019-03-06
MyBatis常见面试题:#{}和${}的区别是什么?
2019-03-06
Vue 数组和对象更新,但视图未更新,背后的故事
2019-03-06
剑指Offer面试题:9.二进制中1的个数
2019-03-06
《你是在做牛做马还是在做主管》- 读书笔记
2019-03-06
ASP.NET Core on K8S学习之旅(12)Ingress
2019-03-06
重新温习软件设计之路(4)
2019-03-06
《刷新》:拥抱同理心,建立成长型思维
2019-03-06
MVC3+NHibernate项目实战(二) :数据库访问层
2019-03-06