UESTC 880 生日礼物

单调队列优化DP。

定义dp[i][j]表示第i天手中有j股股票时,获得的最多钱数。

转移方程有:

1.当天不买也不卖: dp[i][j]=dp[i-1][j];

2.当天买了j-k股: dp[i][j]=max(dp[r][k]+(j-k)*Ap[i]); (r<i-w)

3.当天卖了k-j股: dp[i][j]=max(dp[r][k]+(k-j)*Bp[i]); (r<i-w)

直接转移复杂度太高,为O(n^2*Maxp^2).

分别考虑每种转移,第一种不用管,考虑第二种。

dp[i][j]=max(dp[r][k]+(j-k)*Ap[i])

变换得:dp[i][j]-j*Ap[i]=dp[r][k]-k*Ap[i]。

所以变成使dp[r][k]-k*Ap[i]最大。

对于dp[r][k]-k*Ap[i], 因为我们已经有dp[i][k]=dp[i-1][k]的转移了,说明,dp[i][k]包含了所有dp[j][k]
(j<i)的情况。 换句话说dp[i][k]是递增的。那么这里我们显然就可以直接把r换成i-w-1,于是变成了求
dp[i-w-1][k]-k*Ap[i]的最大值。

令f[k]=dp[i-w-1][k]-k*Ap[i]) 原式变为 dp[i][j]=max(f[k])+j*Ap[i]
(0=<k<j),如果再把与i相关的东西变成常数,则变成类似dp[j] = max(f[k])+c[j]形式,即变成可用单调队列优化的形式。

维护一个单调递增队列来求f[k]。复杂度O(n*Maxp).

因为直接令r=i-w-1,因为r>=1,所以i>w+1时才能转移,这是i<=w+1的情况需要预处理。

代码:

#include <iostream>
#include <cstdio>
#include <cstring>
#include <cmath>
#include <algorithm>
#define Mod 1000000007
using namespace std;
#define N 2007

struct node
{
int num,val;
}que[N];

int AP[N],BP[N],AS[N],BS[N];
int dp[N][N];
int n,Maxp,w;

void init()
{
int i,j;
for(i=0;i<=2000;i++)
for(j=0;j<=Maxp;j++)
dp[i][j] = -Mod;
dp[0][0] = 0;
for(i=1;i<=w+1;i++)
for(j=0;j<=min(AS[i],Maxp);j++)
dp[i][j] = -j*AP[i];
}

int main()
{
int i,j,k;
int t,head,tail;
scanf("%d",&t);
while(t--)
{
scanf("%d%d%d",&n,&Maxp,&w);
for(i=1;i<=n;i++)
scanf("%d%d%d%d",&AP[i],&BP[i],&AS[i],&BS[i]);
init();
for(i=1;i<=n;i++)
{
//unbuy & unsell
for(j=0;j<=Maxp;j++)
dp[i][j] = max(dp[i][j],dp[i-1][j]);
if(i-w-1 <= 0)
continue;
//buy j-k stocks
head = 1;
tail = 0;
for(j=0;j<=Maxp;j++)
{
int tmp = dp[i-w-1][j] + j*AP[i];
while(tail >= head && j-que[head].num > AS[i])
head++;
if(head <= tail)
dp[i][j] = max(dp[i][j],que[head].val-(j-que[head].num)*AP[i]);
while(tail >= head && que[tail].val+que[tail].num*AP[i] < tmp)
tail--;
que[++tail].num = j;
que[tail].val = dp[i-w-1][j];
}
//sell k-j stocks
head = 1;
tail = 0;
for(j=Maxp;j>=0;j--)
{
int tmp = dp[i-w-1][j] + j*BP[i];
while(tail >= head && que[head].num-j > BS[i])
head++;
if(head <= tail)
dp[i][j] = max(dp[i][j],que[head].val-(j-que[head].num)*BP[i]);
while(tail >= head && que[tail].val+que[tail].num*BP[i] < tmp)
tail--;
que[++tail].num = j;
que[tail].val = dp[i-w-1][j];
}
}
int res = 0;
for(i=0;i<=Maxp;i++)
res = max(res,dp[n][i]);
printf("%d\n",res);
}
return 0;
}

UESTC 880 生日礼物

时间: 2024-10-14 22:08:23

UESTC 880 生日礼物的相关文章

UESTC 电子科大专题训练 数据结构 D

UESTC 1584 题意:平面坐标上有n个怪物,每个怪物有一个rank值,代表x坐标和y坐标都不大于它本身的怪物数(不包括本身) 思路:对x y坐标从小到大排序,x优先排序,用数状数组计算y坐标小于它的数量 AC代码: #include "iostream" #include "string.h" #include "stack" #include "queue" #include "string" #i

UESTC 31 饭卡(Card) --背包问题

背包问题. 思路:如果m<5,此时也不能消费,所以此时答案为m m>=5: 求出背包容量为m-5,买前n-1样便宜的菜(排个序)的最大价值(即最大消费,即消费完后剩余值最接近5)最后减去最大的那个菜的价格,就得到最小的余额. 代码: #include <iostream> #include <cstdio> #include <cstring> #include <cmath> #include <algorithm> using n

UESTC 电子科大专题训练 数据结构 A

UESTC 1591 题意:求区间极值之差 思路:线段树裸题,不带更新 ACA代码: #include "iostream" #include "string.h" #include "stack" #include "queue" #include "string" #include "vector" #include "set" #include "ma

2017 UESTC Training for Data Structures

2017 UESTC Training for Data Structures A    水,找区间极差,RMQ怼上去. #include<bits/stdc++.h> using namespace std; #pragma comment(linker, "/STACK:102400000,102400000") #define rep(i,a,b) for (int i=a;i<=b;i++) #define per(i,b,a) for (int i=b;i&

1005 生日礼物

1005 生日礼物 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 黄金 Gold 题目描述 Description 9月12日是小松的朋友小寒的生日.小松知道小寒特别喜欢蝴蝶,所以决定折蝴蝶作为给小寒的生日礼物.他来到了PK大学最大的一家地下超市,在超市里,小松找到了n种可以用来折纸的本子.每种类型的本子里有若干不同颜色的纸若干张,当然同种类型的本子一定是完全一样的,而不同种类型的本子不一定完全不一样.他统计了一下,这里总共有n种不同类型的可以用来折纸的本子,每种本子各有bi

UESTC - 878 温泉旅店 二维费用背包问题

http://acm.uestc.edu.cn/#/problem/show/878 设dp[i][j][k]表示在前i个数中,第一个得到的异或值是j,第二个人得到的异或值是k的方案数有多少种. 因为异或后的大小不确定,所以不能压缩数组,但是也不大..可以过. #include <cstdio> #include <cstdlib> #include <cstring> #include <cmath> #include <algorithm>

生日礼物

这个“礼物”送给我最爱也是最爱我的妈妈. 九月十三,今天这个特别的日子,我差点都给忘了,还是妹妹早上打电话给我才恍然大悟>_<,瞧我这记性. 今天是妈的生日!我竟然给忘了,感觉我这日子过的好糊涂啊┮﹏┭ 在我印象中,从小到大我都很少陪妈过生日,而我的大部分生日都是爸妈陪伴度过的.即使没有陪伴,也总是能收到妈妈的生日礼物.但是我却在妈生日这天不能送妈什么,只是打个电话给妈,早上8点打过去的时候当时她还在上班,后来到中午11点多妈才打开手机接到电话她说工作环境太吵了没听到. 妈,我记得,你最爱唱的

ACM:UESTC - 649 括号配对问题 - stack

  UESTC - 649  括号配对问题 Time Limit: 1000MS   Memory Limit: 65535KB   64bit IO Format: %lld & %llu Description 大家都知道算术表达式中,括号必须配对,现在任意给出一个算术表达式,判断其括号是否配对.如果配对,输出Yes,否则输出No. Input 含多组测试数据,输入首先是一个整数T表示测试数据组数(0<T≤300).随后有T行测试数据,长度不超过1000个字符,字符串间不含空格. Out

2015 CCPC D- Pick The Sticks(UESTC 1218) (01背包变形)

http://acm.uestc.edu.cn/#/problem/show/1218 既然二维dp表示不了,就加一维表示是否在边界放置,放置一个,两个.有一个trick就是如果只放一根,那么多长都可以. wa了好多次(囧) 开始因为l[i]/2会出现小数,没注意,把所有的长度都x2就可以解决. 又wa了n次因为没注意j-l[i]时没加判断,为什么不是RE呢!不开心... /********************************************* Memory: 1140 KB