CDOJ 1132 酱神赏花 dp+单调栈降低复杂度+滚动数组

酱神赏花

Time Limit: 3000/1000MS (Java/Others)     Memory Limit: 262143/262143KB (Java/Others)

Submit Status

酱神去杭州赏花。

花展在一条街道上举行,这条街道上有一共有n个节点,自左而右从11到nn编号,11号和nn号是左右两个端点,两个相邻端点之间的距离为11.本次花展一共要展出mm朵花,在第titi时刻,有一朵颜值为bibi的花将在第aiai个节点展出,如果酱神在titi时刻处于第xx个节点,那么他能获得的开心值为bi−|x−ai|bi−|x−ai|,注意这个值可能为负。

在t=1t=1的时刻,酱神可以随意从11到nn选出一个节点作为赏花的起点。在接下来的每个单位时间段中,酱神最多能移动dd的距离。酱神每秒只能移动整数个距离,且任何时刻不能超出街道的范围。

他能获得的最大开心值为多少?

Input

第一行33个数n,m,dn,m,d。

接下来mm行,每行33个数ai,bi,tiai,bi,ti。

1≤n≤1051≤n≤105,1≤m≤1001≤m≤100

1≤ai≤n1≤ai≤n

1≤bi≤1091≤bi≤109

1≤ti≤1091≤ti≤109

1≤d≤1091≤d≤109

Output

输出一个数,酱神的最大开心值。

Sample input and output

Sample Input Sample Output
30 4 2
27 3 1
11 4 1
11 4 1
1 2 20
-3

Source

2015 UESTC Training for Dynamic Programming

题意:在数轴上有1-n的n个点,共有m多花要开放,每次会在ti时刻在ai位置开出一朵颜值为bi的花

假如花开时你在x处,那么你得到的颜值是bi-(ai-x),你单位时间移动的速度为d,为你能获得的最大颜值。

#include <iostream>
#include <cstdio>
#include <cstring>
#include <cstdlib>
#include <cmath>
#include <vector>
#include <queue>
#include <stack>
#include <map>
#include <algorithm>
#include <set>
using namespace std;
typedef long long ll;
typedef unsigned long long Ull;
#define MM(a,b) memset(a,b,sizeof(a));
const double eps = 1e-10;
const int  inf =0x7f7f7f7f;
const double pi=acos(-1);
const int N=100005;

struct node{
   int a,b;ll t;
}flower[N];

bool cmp(node a,node b)
{
   return a.t<b.t;
}

ll dp[N][3];
int n,m,d,x[N];

void init(int cur)
{
    for(int i=1;i<=n;i++) dp[i][cur]=-1e15;
}

int main()
{
    while(~scanf("%d%d%d",&n,&m,&d))
    {
        for(int i=1;i<=m;i++)
            scanf("%d%d%lld",&flower[i].a,&flower[i].b,&flower[i].t);
        sort(flower+1,flower+m+1,cmp);

        int cur=0;
        for(int i=1;i<=n;i++)
            {
                dp[i][cur]=flower[1].b-abs(flower[1].a-i);
            }

        for(int k=2;k<=m;k++)
        {
            cur^=1;
            init(cur);
            ll limit=(flower[k].t-flower[k-1].t)*d;
            int l=1,r=1;
            for(int i=1;i<=n;i++)
            {
                while(l<r&&dp[i][cur^1]>=dp[x[r-1]][cur^1])  r--;
                x[r]=i;r++;
                while(l<r&&(i-x[l]>limit))  l++;
                dp[i][cur]=max(dp[i][cur],dp[x[l]][cur^1]+flower[k].b-abs(flower[k].a-i));
            }

            l=1;r=1;
            for(int i=n;i>=1;i--)
            {
                while(l<r&&dp[i][cur^1]>=dp[x[r-1]][cur^1])  r--;
                x[r]=i;r++;
                while(l<r&&(x[l]-i>limit))  l++;
                dp[i][cur]=max(dp[i][cur],dp[x[l]][cur^1]+flower[k].b-abs(flower[k].a-i));
            }
        }

        ll ans=-1e15;
        for(int i=1;i<=n;i++) ans=max(ans,dp[i][cur]);
        printf("%lld\n",ans);
    }
    return 0;
}

  分析:经典的一道题,

1.首先预处理出第一次花开时,枚举在1-n各个点能获得的颜值。

2.dp[i][k]即为第k次花开时,人在i处获得的最大颜值,那么从小到大枚举每个i,并且维护一个从左到右的单调递减栈(第k-1多花开时),再判断当前枚举的i在单调栈中能达到的最左的点(限制条件就是人转移的距离),然后再从最左点转移一下到当前枚举的点i;

3.最后因为一个点可以从左边转移过来,也可以从右边转移过来,所以需要i从n到1再进行一次2步骤

这道题目能用单调栈降下来,关键是因为加入i跟i-1都能转移到j位置,那么如果dp[i][k-1]>=dp[i-1][k-1]的话,那么i-1这个点就不要判断了,直接丢掉

时间: 2024-12-26 02:45:09

CDOJ 1132 酱神赏花 dp+单调栈降低复杂度+滚动数组的相关文章

UESTC_酱神赏花 2015 UESTC Training for Dynamic Programming&lt;Problem C&gt;

C - 酱神赏花 Time Limit: 3000/1000MS (Java/Others)     Memory Limit: 262143/262143KB (Java/Others) Submit Status 酱神去杭州赏花. 花展在一条街道上举行,这条街道上有一共有n个节点,自左而右从1到n编号,1号和n号是左右两个端点,两个相邻端点之间的距离为1.本次花展一共要展出m朵花,在第ti时刻,有一朵颜值为bi的花将在第ai个节点展出,如果酱神在ti时刻处于第x个节点,那么他能获得的开心值为

【bzoj4540】[Hnoi2016]序列 单调栈+离线+扫描线+树状数组区间修改

题目描述 给出一个序列,多次询问一个区间的所有子区间最小值之和. 输入 输入文件的第一行包含两个整数n和q,分别代表序列长度和询问数.接下来一行,包含n个整数,以空格隔开,第i个整数为ai,即序列第i个元素的值.接下来q行,每行包含两个整数l和r,代表一次询问. 输出 对于每次询问,输出一行,代表询问的答案. 样例输入 5 5 5 2 4 1 3 1 5 1 3 2 4 3 5 2 5 样例输出 28 17 11 11 17 题解 单调栈+离线+扫描线+树状数组区间修改 首先把使用单调栈找出每个

cdoj 1141 酱神寻宝 状压dp

酱神寻宝 Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://acm.uestc.edu.cn/#/problem/show/1141 Description 酱神来到了一座小岛,岛上有n个箱子. 一共有3中不同的钥匙,金钥匙.银钥匙和万能钥匙.酱神一开始有a把金钥匙.b把银钥匙和c把万能钥匙. 第i个箱子上有xi把金锁,yi把银锁.金钥匙只能打开金锁,银钥匙只能打开银锁,万能钥匙两种锁都能打开.用于打开锁的钥匙会立刻损坏,酱神会丢掉损坏的钥匙

【DP/单调栈】关于单调栈的一些题目(codevs 1159,codevs 2673)

题目描述 Description 这个月的pku月赛某陈没有参加,因为当时学校在考试[某陈经常逃课,但某陈还没有强大到考试也可以逃掉的程度].何况,对于北大校赛,水牛通常是没有什么希望考得好的[事实上某陈最好成绩是仅A了一道题]. 某陈郁闷.接下来他又将沉浸在无穷尽的刷题中,每天面对各种颜色的Status--WA,TLE,RE,甚至还有MLE,CE,PE什么什么的,他无比期待蓝色的AC. 话说RP爆发的某陈弄到了很久以后某次pku月赛的某题的题目和输入数据,如下所示. 输入数据包括n个测试点,每

G - Most Dangerous Shark dp 单调栈 单调队列

题意: 一条线段上有n张骨牌(n=1e7), 相邻骨牌距离为1,每张骨牌有其高度和推倒的花费,问最少的花费推倒所有的骨牌. 题解: 首先用单调栈维护每个位置往左(右)推能推倒的最远的骨牌 dp[i]表示1-i倒下的最小花费 转移显然只有两种  一种是第i张往左推动 另一种是找到往右推动能推倒i的最小花费的地方进行转移  这里可以用单调栈维护最小值 #include<bits/stdc++.h> using namespace std; typedef long long ll; const i

CDOJ 1069 秋实大哥去打工 单调栈 下标处理

E - 秋实大哥去打工 Time Limit:1000MS     Memory Limit:65535KB     64bit IO Format:%lld & %llu Submit Status Practice UESTC 1069 Appoint description:  System Crawler  (2016-04-24) Description 天行健,君子以自强不息.地势坤,君子以厚德载物. 天天过节的秋实大哥又要过节了,于是他要给心爱的妹子买礼物.但由于最近秋实大哥手头拮

poj1159--Palindrome(dp:最长公共子序列变形 + 滚动数组)

Palindrome Time Limit: 3000MS   Memory Limit: 65536K Total Submissions: 53414   Accepted: 18449 Description A palindrome is a symmetrical string, that is, a string read identically from left to right as well as from right to left. You are to write a

bzoj1057: [ZJOI2007]棋盘制作 [dp][单调栈]

Description 国际象棋是世界上最古老的博弈游戏之一,和中国的围棋.象棋以及日本的将棋同享盛名.据说国际象棋起源 于易经的思想,棋盘是一个8*8大小的黑白相间的方阵,对应八八六十四卦,黑白对应阴阳.而我们的主人公小Q, 正是国际象棋的狂热爱好者.作为一个顶尖高手,他已不满足于普通的棋盘与规则,于是他跟他的好朋友小W决定 将棋盘扩大以适应他们的新规则.小Q找到了一张由N*M个正方形的格子组成的矩形纸片,每个格子被涂有黑白两种 颜色之一.小Q想在这种纸中裁减一部分作为新棋盘,当然,他希望这个

ABC 158 F - Removing Robots dp 单调栈

LINK:Removing Robots 没想到 自闭. 考虑了一个容斥 发现不合法方案难以计算. 就算可以计算也几乎是n^2的做法. 考虑dp 左边会对右边产生影响 所以考虑先dp右边的再考虑左边的. 至于dp 自然是f[i]表示这个i~n这么多点的方案数. 设 i向右第一个影响不到的点为w 那么 有转移 f[i]+=f[i+1]+f[w]. 这样从左到右dp就可以算出所有的方案了. const ll MAXN=200010; ll n,ans,top; ll f[MAXN],g[MAXN],