洛谷P1860——新魔法药水
传送门:QAQQAQ
题意:商店里有N种药水,每种药水都有一个售价和回收价。小S攒了V元钱,还会M种魔法,可以把一些药水合成另一种药水。他一天可以使用K次魔法,问他一天最多赚多少钱?
N<=60 M<=240
V<=1000
k<=30
思路:这是一道比较有技术含量的DP题。
我们定义$dp[i][j]$为消耗了$i$个金币(金币不能回收),使用了$j$次魔法能得到的最多钱(最后求答案时注意要$dp[i][j]-i$)
在直接转移的过程中,我们无法知道使用的金币都花在了哪些药品上,各种魔法又都用了几次,直接枚举又不知如何下手,所以我们要定义辅助数组:
我们定义$cost[i][j]$为最终消耗了$j$次魔法,得到了药品$i$的最小花费,这样转移方程就很容易了:
$dp[i][j]=max(dp[i-cost[p][t]][j-t]+w[p])$
$cost[i][j]=min(\sum cost[magic[p].to[t]][r]) (\sum r=j-1)$
而在处理$cost[i][j]$时我们又遇到了一些难题:我们没法知道在得到$i$的魔法中每种原料到底用了几次,其“下层”的魔法各用了几次,所以我们要再开一个辅助数组
我们定义对于当前搜到的魔法$i$,$ant[t][r]$为前该魔法前$t$个物品使用了$r$次魔法的最小花费
我们可以通过$t$逐步增大来更新后面的$ant$,其中$cost$和$ant$数组是互相利用的。
(在代码实现方面,要注意$init$函数里的循环顺序,一定要先枚举使用魔法数,这样才能保证在当前枚举到魔法数$j$时前面所有比$j$小的$ant$,$cost$数组已经最优化,这样就可以无忧无虑地转移啦~~)
代码:(用刷表写的,之前刷表越界了。。。)
#include<bits/stdc++.h>
using namespace std;
const int inf=(int)1e9; int cost[][],dp[][],ant[][];
int n,m,v,k;
int b[],s[];
struct node{
int from,len;
int to[];
}E[]; void checkmax(int &x,int y)
{
if(x<y) x=y;
} void checkmin(int &x,int y)
{
if(x>y) x=y;
} void init()
{
for(int i=;i<=n;i++)
{
for(int j=;j<=k;j++) cost[i][j]=b[i];
}
for(int j=;j<=k;j++)
{
for(int i=;i<=m;i++)
{
for(int t=;t<=E[i].len;t++)
{
for(int r=;r<=j-;r++)
{
ant[t][r]=inf;
for(int p=;p<=r;p++) checkmin(ant[t][r],ant[t-][p]+cost[E[i].to[t]][r-p]);
}
}
checkmin(cost[E[i].from][j],ant[E[i].len][j-]);
}
}
} void solve()
{
memset(dp,,sizeof(dp));
for(int i=;i<=v;i++)
{
for(int j=;j<=k;j++)
{
for(int t=;t<=n;t++)//新加药水种类
{
for(int p=;p<=k-j;p++)//新用魔法次数
{
if(i+cost[t][p]>v||j+p>k) continue;
checkmax(dp[i+cost[t][p]][j+p],dp[i][j]+s[t]);
}
}
}
}
int ans=;
for(int i=;i<=v;i++)
{
for(int j=;j<=k;j++) checkmax(ans,dp[i][j]-i);
}
cout<<ans<<endl;
} int main()
{
scanf("%d%d%d%d",&n,&m,&v,&k);
for(int i=;i<=n;i++) scanf("%d%d",&b[i],&s[i]);
for(int i=;i<=m;i++)
{
scanf("%d",&E[i].from);
scanf("%d",&E[i].len);
for(int j=;j<=E[i].len;j++) scanf("%d",&E[i].to[j]);
}
init();
solve();
return ;
}
洛谷P1860——新魔法药水的更多相关文章
- 洛谷P1860 新魔法药水
洛谷题目链接 动态规划: 这个题目调了我好久....结果循环变量写错了... 而且题目有个坑!!!只能用开始给你的$v$元买入东西 回归正题: 我们定义状态$ans[i][j]$表示第$i$个物品用了 ...
- 洛谷P1242 新汉诺塔(dfs,模拟退火)
洛谷P1242 新汉诺塔 最开始的思路是贪心地将盘子从大到小依次从初始位置移动到目标位置. 方法和基本的汉诺塔问题的方法一样,对于盘子 \(i\) ,将盘子 \(1\to i-1\) 放置到中间柱子上 ...
- 洛谷 P1305 新二叉树
P1305 新二叉树 题目描述 输入一串完全二叉树,用遍历前序打出. 输入输出格式 输入格式: 第一行为二叉树的节点数n. 后面n行,每一个字母为节点,后两个字母分别为其左右儿子. 空节点用*表示 输 ...
- Luogu P1860 新魔法药水
题目大意 具体题面及输入格式戳我! 商店里有\(N\)种药水,每种药水都有一个售价和回收价. 小\(S\) 攒了\(V\)元钱,还会\(M\)种魔法,可以把一些药水合成另一种药水. 他在第一天可以购买 ...
- 洛谷 P1305 新二叉树 Label:字符串的输出总是有惊喜
题目描述 输入一串完全二叉树,用遍历前序打出. 输入输出格式 输入格式: 第一行为二叉树的节点数n. 后面n行,每一个字母为节点,后两个字母分别为其左右儿子. 空节点用*表示 输出格式: 前序排列的完 ...
- 洛谷P1242 新汉诺塔
传送门啦 首先要将第n个盘子从x到y,那么就要把比n小的盘子全部移到6-x-y,然后将n移到y 仔细想想:6代表的是3根初始柱,3根目标柱. 6-(x+y) 便是我们的中转柱了,因为到这个位置是最优的 ...
- 洛谷P1242 新汉诺塔 【神奇的递归】
题目描述 设有n个大小不等的中空圆盘,按从小到大的顺序从1到n编号.将这n个圆盘任意的迭套在三根立柱上,立柱的编号分别为A.B.C,这个状态称为初始状态. 现在要求找到一种步数最少的移动方案,使得从初 ...
- [P1860]新魔法药水
题目描述 商店里有N种药水,每种药水都有一个售价和回收价.小S攒了V元钱,还会M种魔法,可以把一些药水合成另一种药水.他一天可以使用K次魔法,问他一天最多赚多少钱? 输入输出格式 输入格式: 第一行四 ...
- 洛谷 [P4301] 新Nim游戏
线性基 +博弈论 先手必胜当且仅当先手取完之后留下的序列无论如何组合,异或和都不为 0 也就是剩下的整数线性无关,所以我们对所有整数排序,由高往低的贪心的插入线性基, 无法插入的就有先手取出,容易发现 ...
随机推荐
- 【Dart学习】-- Dart之操作符
一,概述 dart定义了下表所示的运算符.你可以重写许多这些运算符. 描述 运算符 一元后缀 expr++ expr-- () [] . ?. 一元前缀 -expr !expr ~expr ++exp ...
- html标签中的 MIME 类型
1.<script type="text/x-template"> MIME 类型 2.MIME类型是什么 : http://www.w3school.com.cn/ ...
- [SDOI2010]地精部落 题解
Description 传说很久以前,大地上居住着一种神秘的生物:地精. 地精喜欢住在连绵不绝的山脉中.具体地说,一座长度为 N 的山脉 H可分 为从左到右的 N 段,每段有一个独一无二的高度 Hi, ...
- ios获取软键盘完成事件
ios获取软键盘完成事件,通过判断input的onBlur事件即可
- 使用Netfilter进行数据包分析
#include <linux/init.h>#include <linux/module.h>#include <linux/skbuff.h>#include ...
- linux内核编译时如何根据spec指定编译包
问题: 1> rpmbuild -bb SPECS/kernel.spec --define="_topdir `pwd`" 编译 出的包并未包含kernel-firmwar ...
- 在ubuntu16下安装virtualenv+virtualenvwrapper
ubuntu16已经安装好了py2和py3(自带的) bigni@bigni-Latitude-E6230:~/python_file/Django_project$ python python py ...
- URL&HTTP协议&GET请求&POST请求
1.什么是URL URL的全称是Uniform Resource Locator(统一资源定位符) 通过1个URL,能找到互联网上唯一的1个资源 URL就是资源的地址.位置,互联网上的每个资源都有一个 ...
- Flyway 学习时遇到的错误
错误一: No plugin found for prefix 'flyway' in the current project and in the plugin groups 找不到Flyway插 ...
- ES6数组Api扩充
1. Array.of( ); ----将一组数据转换成一个数组: const num=201314; const a=Array.of(num); console.log(a); //数组 ...