fjnu2019第二次友谊赛 F题
题目大意:
一开始手上有 z 个钱币,有 n 天抉择,m 种投资方案,在每天中可以选择任意种投资方案、任意次地花费 x 个钱币(手上的钱币数不能为负),使得在 n 天结束后,获得 y 个钱币。
其次,在每天结束后,会根据自己手上所具有的节点数来获得一些钱币补偿,设当天结束后所拥有 x 个钱币,那么将获得 f(x) 个钱币,若 x > k ,则默认为 0 。保证 f[0]~f[k] 单调不增。
这题的原题:博客连接 ,只是一开始手上的钱数不同而已。
分析:
1、将全过程分为 n 天,在 dp 枚举每天的状态时,先处理当天一开始所拥有的钱币数(即上一天结束后手上的钱数),再进行当天的投资。所以我们将当天投资结束后的补偿状态在下一天的早上进行转移,而不是在当天结束后进行状态转移。
2、故设 dp[i][j] 表示在第 i 天投资完之后(当天还未获得补偿),在手上拥有 j 个钱币时,在 n 天结束后所获得的返还的最大钱币总数。
3、那么对于每一天的一开始钱币数是由上一天补偿之后而转移过来的,由于补偿只会在有剩下 0 ~ k 个钱币时才会发生 (即 手上钱币数为 0 ~ k 时才可能获得补偿),故在每天开始前需要遍历上一天手中剩下的钱币数,获得一定的补偿之后,成为今天一开始的钱币数。
4、获得当天的钱币数后,开始进行今天的投资状态转移。由于投资的数量为任意次,故为完全背包的转移。
5、若对于当天投资结束后手上有 j 元时的状态( 即 dp[i][j] ),它必从当天投资一开始的手上钱币为 ( j + 投资成本 )时,花费投资成本后转移而来。故若设投资成本为 w ,最后一天返还 v ,则有 dp[i][j] = dp[i][j + w] + v ,完全背包取最大值即可。
6、由于一开始有钱币数,而按上面所述,第一天一开始不能从上一天剩余钱币数上转移,故第一天需要单独拿出来先处理。
7、初始化问题:按理这题 dp 需要求最大值,全部初始化为 0 才对,但是需要保证本题的实际意义——需要从第一天手上 z 个钱币时转移,故需要初始化全部为负无穷,然后将dp[1][z] 赋值为 0 ,表明 dp 转移时,必须从第一天手上有 z 个钱币时转移而来。
8、由于我们将第一天枚举单独拿出来了,即意味着求的所有状态都必须在第一天买东西才会被转移到第二天(即第一天啥都不买时,这个状态不会被传递下去),然后再进行下面的每天转移。故我们求答案时,还需要判断是否 n 天啥都不投资的钱币会更多(意思是如果第一天啥都不买,最大值不可能轮到第二天才开始买)。
注意点:
1、若按上述这样直接做的话,复杂度接近 n³ * k (k 为常数),此题数据将会有 3000MS (当然出题人良心放宽时限)。
2、若要降低时间复杂度,需要知道完全背包去掉枚举方案个数的原理。这个原理其实是当前 dp[i][j] 从本层之前已更新过的状态转移过来(详细则自己百度或群里问)。
3、此题与普通完全背包降维不同的是,分析 dp 转移方程,发现他是从后往前转移的 (即 dp[i][j] 中的 j 从 j + w 上转移而来),优化时,需要反向枚举背包容量。
无优化代码:
#include<iostream>
#include<algorithm>
#include<string.h>
using namespace std;
int z,N,M,K;
int dp[][];
int f[];
struct Goods{
int a,b;
}A[];
int main()
{
scanf("%d%d%d%d",&z,&N,&M,&K);
for(int i=;i<=K;i++) scanf("%d",&f[i]);
for(int i=;i<=M;i++) scanf("%d%d",&A[i].a,&A[i].b);
memset(dp,0x80,sizeof(dp));
dp[][z]=;
for(int w=;w<=M;w++){
for(int j=A[w].a;j<=;j++){
for(int k=;k<=j/A[w].a;k++){
dp[][j-k*A[w].a]=max(dp[][j-k*A[w].a],dp[][j]+k*A[w].b);
}
}
}
for(int i=;i<=N;i++){
for(int j=;j<=K;j++) dp[i][j+f[j]]=max(dp[i][j+f[j]],dp[i-][j]);
for(int w=;w<=M;w++){
for(int j=A[w].a;j<=;j++){
for(int k=;k<=j/A[w].a;k++){
dp[i][j-k*A[w].a]=max(dp[i][j-k*A[w].a],dp[i][j]+k*A[w].b);
}
}
}
}
int ans=z;
for(int i=;i<=;i++) ans=max(ans,dp[N][i]+i+f[i]);
printf("%d\n",ans);
}
优化代码:
#include<iostream>
#include<algorithm>
#include<string.h>
using namespace std;
int z,N,M,K;
int dp[][];
int f[];
struct Goods{
int a,b;
}A[];
int main()
{
scanf("%d%d%d%d",&z,&N,&M,&K);
for(int i=;i<=K;i++) scanf("%d",&f[i]);
for(int i=;i<=M;i++) scanf("%d%d",&A[i].a,&A[i].b);
memset(dp,0x80,sizeof(dp));
dp[][z]=;
for(int w=;w<=M;w++){
for(int j=;j>=A[w].a;j--){
dp[][j-A[w].a]=max(dp[][j-A[w].a],dp[][j]+A[w].b);
}
}
for(int i=;i<=N;i++){
for(int j=;j<=K;j++) dp[i][j+f[j]]=max(dp[i][j+f[j]],dp[i-][j]);
for(int w=;w<=M;w++){
for(int j=;j>=A[w].a;j--){
dp[i][j-A[w].a]=max(dp[i][j-A[w].a],dp[i][j]+A[w].b);
}
}
}
int ans=z;
for(int i=;i<=;i++) ans=max(ans,dp[N][i]+i+f[i]);
printf("%d\n",ans);
}
fjnu2019第二次友谊赛 F题的更多相关文章
- fjnu2019第二次友谊赛 B题
### 题目链接 ### 题目大意: 给你一个 n * m 的地图以及小蛇蛇头的初始位置,告诉你它会往 上.下.左.右 四个方向走.若在走的过程中(包括结束时)会使得小蛇越界,则输出 "Ga ...
- 2013年山东省赛F题 Mountain Subsequences
2013年山东省赛F题 Mountain Subsequences先说n^2做法,从第1个,(假设当前是第i个)到第i-1个位置上哪些比第i位的小,那也就意味着a[i]可以接在它后面,f1[i]表示从 ...
- 2017Summmer_上海金马五校 F题,G题,I题,K题,J题
以下题目均自己搜 F题 A序列 一开始真的没懂题目什么意思,还以为是要连续的子串,结果发现时序列,简直智障,知道题意之后,好久没搞LIS,有点忘了,复习一波以后,直接双向LIS,处理处两个数组L和R ...
- ACM-ICPC 2019南昌网络赛F题 Megumi With String
ACM-ICPC 南昌网络赛F题 Megumi With String 题目描述 给一个长度为\(l\)的字符串\(S\),和关于\(x\)的\(k\)次多项式\(G[x]\).当一个字符串\(str ...
- 2019牛客多校第八场 F题 Flowers 计算几何+线段树
2019牛客多校第八场 F题 Flowers 先枚举出三角形内部的点D. 下面所说的旋转没有指明逆时针还是顺时针则是指逆时针旋转. 固定内部点的答案的获取 anti(A)anti(A)anti(A)或 ...
- AtCoder Beginner Contest 215 F题题解
F - Dist Max 2 什么时候我才能突破\(F\)题的大关... 算了,不说了,看题. 简化题意:给定\(n\)个点的坐标,定义没两个点的距离为\(min(|x_i-x_j|,|y_i-y_j ...
- 2019年牛客多校第二场 F题Partition problem 爆搜
题目链接 传送门 题意 总共有\(2n\)个人,任意两个人之间会有一个竞争值\(w_{ij}\),现在要你将其平分成两堆,使得\(\sum\limits_{i=1,i\in\mathbb{A}}^{n ...
- (中等) Hiho 1232 Couple Trees(15年北京网络赛F题),主席树+树链剖分。
"Couple Trees" are two trees, a husband tree and a wife tree. They are named because they ...
- AtCoder Beginner Contest 213 F题 题解
F - Common Prefixes 该题也是囤了好久的题目了,看题目公共前缀,再扫一眼题目,嗯求每个后缀与其他后缀的公共前缀的和,那不就是后缀数组吗?对于这类问题后缀数组可是相当在行的. 我们用后 ...
随机推荐
- cacti,zabbix,nagios各自原理及区别
cacti,zabbix,nagios各自原理及区别 Taxing祥 0人评论 19462人阅读 2017-09-24 00:23:54 cacti原理: 1,构件(步骤): net_snmp:负 ...
- windows10查看电脑已经保存的wifi密码
1,打开windows的命令窗口,输入 netsh wlan show profiles,如下图,这个命令仅仅只是查看一下电脑保存的所有的wifi名字 2,需要查看密码的话,则需要输入这个命令, ...
- 《Java知识应用》Java压缩文件和解压缩
今天通过Java实现一下:文件的压缩和解压缩. 1. 压缩 准备文件: 准备代码:(压缩) import java.io.BufferedInputStream; import java.io.Buf ...
- conda docker镜像
之前的python环境,使用ubuntu安装pip来安装python依赖,但是遇到缺少某些库的版本,比如一个项目需要用到faiss,pip只有最新的1.5.3版本,但是这个版本使用了较新的CPU指令, ...
- Linux下的密码破解
密码散列: 密码散列的$6 表示是:SHA512 这里我们使用hashcat 工具进行破解 ╰─ hashcat -m 1800 hash.txt /usr/share/wordlists/rocky ...
- MESSAGE_TYPE_X dump in RSM_DATASTATE_CHECK -6-
DTP抽数时系统Dump 参考sapnote:2398760 - MESSAGE_TYPE_X dump in RSM_DATASTATE_CHECK -1- to -12- RSM_DATASTAT ...
- Mac 安装nginx之后重启、停止、开启等操作
操作系统:macOs High Sierra 10.13.6 1.我用的homebrew安装的nignx1.15.9,安装完成之后会有下面的提示: 网站根目录在:/usr/local/var/www ...
- 对《Java核心技术卷一》读者的一些建议
<Java核心技术卷一>是唯一可以和<Java编程思想>媲美的一本 Java 入门书.单从技术的角度来看,前者更好一些.但上升到思想层面嘛,自然后者更好,两者的偏重点不同. 思 ...
- Prometheus学习系列(三)之Prometheus 概念:数据模型、metric类型、任务、实例
前言 本文来自Prometheus官网手册1.Prometheus官网手册2 和 Prometheus简介 说明 Prometheus从根本上存储的所有数据都是时间序列: 具有时间戳的数据流只属于单个 ...
- js 实现ReplaceAll 的方法
JS 字符串有replace() 方法.但这个方法只会对匹配到的第一个字串替换. 如下例: <HTML> <HEAD> <TITLE> New Document ...