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 该题也是囤了好久的题目了,看题目公共前缀,再扫一眼题目,嗯求每个后缀与其他后缀的公共前缀的和,那不就是后缀数组吗?对于这类问题后缀数组可是相当在行的. 我们用后 ...
随机推荐
- 小胖求学系列之-文档生成利器(下)-smart-doc
叮叮叮....一阵铃声响起,大家都陆续来到了课堂,看老师没来,小张和小胖又闲聊起来,小张问:怎么样,smart-doc好用吧.小胖笑着说:挺好用的,不过? 小张看卖关子,问到:不过什么,有什么新发现? ...
- 【Feign】自定义配置
[Feign]自定义配置 转载: 自定义配置,如果在同一个工程,注意配置不要和@SpringBootApplication或@ComponentSacan放在用一个包下,就是不要被扫描上 packag ...
- 更新Preloader和uboot
在SoCEDS环境下编译和更新preloader和uboot程序的方法 前面有介绍preloader在HPS boot过程中的的作用,接下来讲述下用户在SoCEDS环境下改如何编译preloade ...
- 微信小程序 wepy框架 之拦截器intercepter使用
1,在使用wepy框架创建的项目下 找到src/app.wpy 2,在app.wpy constructor方法中添加 super(); this.use('promisify');//启用ES6 p ...
- spring+activemq实战之配置监听多队列实现不同队列消息消费
摘选:https://my.oschina.net/u/3613230/blog/1457227 摘要: 最近在项目开发中,需要用到activemq,用的时候,发现在同一个项目中point-to-po ...
- C# 几种常见数据结构
一.内存上连续存储,节约空间,可以索引访问,读取快,增删慢 Array:在内存上连续分配的,而且元素类型是一样的,可以坐标访问;读取快--增删慢,长度不变 { //Array:在内存上连续分配的,而且 ...
- 如何正确使用 Spring Cloud?【中】
3. Spring 集成了哪些常用组件? 从 2004 年发布 1.0 版本开始,Spring 目前已经演进至 5.x 版本了,为不同时期的应用开发提供了强有力的支撑.现在我们正面对微服务.DevOp ...
- SVN中怎样忽略当前文件不提交
场景 在使用SVN进行版本管理时,有时一些自动生成的文件比如证书等,在每台电脑上都会不同,如果将其提交,则会冲突. 怎样将指定的文件或者指定文件后缀的文件忽略提交. 注: 博客主页: https:// ...
- 渗透测试初学者的靶场实战 3--墨者学院SQL注入—宽字节盲注
墨者SQL注入-MYSQL数据库实战环境 实践步骤 1. 决断注入点 输入单引号,提示错误信息: 输入and 1=1 返回页面正常: 输入 and 1=2 返回正常 输入-1,返回异常: 2. 带入s ...
- JDK1.8新特性-Lambda表达式
Lambda 表达式 Lambda 表达式,也可称为闭包,它是推动 Java 8 发布的最重要新特性. Lambda 允许把函数作为一个方法的参数(函数作为参数传递进方法中). 使用 Lambda 表 ...