【NOIP数据结构专项】单调队列单调栈

【洛谷P1901 】发射站

http://www.luogu.org/problem/show?pid=1901

题目描述

某地有 N 个能量发射站排成一行,每个发射站 i 都有不相同的高度 Hi,并能向两边(当 然两端的只能向一边)同时发射能量值为 Vi 的能量,并且发出的能量只被两边最近的且比 它高的发射站接收。

显然,每个发射站发来的能量有可能被 0 或 1 或 2 个其他发射站所接受,特别是为了安 全,每个发射站接收到的能量总和是我们很关心的问题。由于数据很多,现只需要你帮忙计 算出接收最多能量的发射站接收的能量是多少。

输入格式

第 1 行:一个整数 N;

第 2 到 N+1 行:第 i+1 行有两个整数 Hi 和 Vi,表示第 i 个人发射站的高度和发射的能量值。

输出格式

输出仅一行,表示接收最多能量的发射站接收到的能量值,答案不超过 longint。

输入样例:

3
4 2
3 5
6 10

输出样例:

7

数据规模

对于 40%的数据,1<=N<=5000;1<=Hi<=100000;1<=Vi<=10000;

对于 70%的数据,1<=N<=100000;1<=Hi<=2,000,000,000;1<=Vi<=10000;

对于 100%的数据,1<=N<=1000000;1<=Hi<=2,000,000,000;1<=Vi<=10000。

题解

单调栈维护找出向左(右)边的第一个比自己大的数。

#include<iostream>
#include<cstdlib>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<algorithm>
#include<stack>
using namespace std;

long long h[1000001],p[1000001],sum[1000001];
struct hh{int l,r;};
hh f[1000001];
long long ans;
int n,cnt;
stack<int> s;

int main()
{
    int i,j;
    scanf("%d",&n);
    for(i=1;i<=n;i++)
      scanf("%lld%lld",&h[i],&p[i]);
    s.push(1);
    for(i=2;i<=n;++i)
    {
        while(!s.empty() && h[i]>h[s.top()]) f[s.top()].r=i,s.pop();
        s.push(i);
    }
    while(!s.empty()) s.pop();
    s.push(n);
    for(i=n-1;i>=1;--i)
    {
        while(!s.empty() && h[i]>h[s.top()]) f[s.top()].l=i,s.pop();
        s.push(i);
    }

    for(i=1;i<=n;i++){sum[f[i].l]+=p[i]; sum[f[i].r]+=p[i];}
    for(i=1;i<=n;i++) ans=max(ans,sum[i]);
    printf("%lld",ans);
    return 0;

}

【POJ2823 】【FZYZ P1561】滑动窗口

http://poj.org/problem?id=2823

Description

给你一个长度为N的数组,一个长为K的滑动的窗体从最左移至最右端,你只能见到窗口的K个数,每次窗体向右移动一位,如下表:

你的任务是找出窗口在各位置时的max value,min value.

Input Format

第1行n,k,第2行为长度为n的数组

Output Format

2行,第1行每个位置的min value,第2行每个位置的max value

Sample Input
8 3
1 3 -1 -3 5 3 6 7 
Sample Output
-1 -3 -3 -3 3 3
3 3 5 5 6 7 
Hint

20%:n<=500;

50%:n<=100000;

100%:n<=1000000;

题解

双端单调队列维护

#include<iostream>
#include<cstdlib>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<algorithm>
#include<queue>
using namespace std;
int n,k;
int a[1000005],maxx[1000005],minn[1000005];
int q[1000005];

int main()
{
    int i,j,head=1,tail=0;
    scanf("%d%d",&n,&k);
    for(i=1;i<=n;i++) scanf("%d",&a[i]);

    head=1;tail=0;q[++tail]=1;
    for(i=2;i<=k;i++)
    {
        while(head<=tail&&a[q[tail]]<a[i]) tail--;
        q[++tail]=i;
    }
    maxx[1]=a[q[head]];
    for(i=k+1;i<=n;i++)
    {
        while(head<=tail&&a[q[tail]]<a[i]) tail--;
        while(head<=tail&&i-q[head]>=k) head++;
        q[++tail]=i;
        maxx[i-k+1]=a[q[head]];
    }

    head=1;tail=0;
    q[++tail]=1;
    for(i=2;i<=k;i++)
    {
        while(head<=tail&&a[q[tail]]>a[i]) tail--;
        q[++tail]=i;
    }
    minn[1]=a[q[head]];
    for(i=k+1;i<=n;i++)
    {
        while(head<=tail&&a[q[tail]]>a[i]) tail--;
        while(head<=tail&&i-q[head]>=k) head++;
        q[++tail]=i;
        minn[i-k+1]=a[q[head]];
    }
    for(i=1;i<=n-k+1;i++) printf("%d ",minn[i]);printf("\n");
    for(i=1;i<=n-k+1;i++) printf("%d ",maxx[i]);
    return 0;
}

【FZYZ P1280 】【NOIP福建夏令营】矩形覆盖

Description

有N个矩形,矩形的底边边长为1,且均在X轴上,高度给出,第i个矩形的高为h[i],求最少需要几个矩形才能覆盖这个图形。

例如h = [3, 2, 4, 2]的图形如下:

容易发现,只需要3个矩形就能覆盖这个图形。

Input Format

第一行一个整数N。接下来1行包含N个正整数,为h[i]。

Output Format

输出一个整数表示最少需要几个矩形能覆盖这个图形。

Sample Input
10
2 3 2 4 2 1 3 4 3 2
Sample Output
7
Hint

对于所有数据,N<=100000,h[i] <= 100。

对于部分数据,N<=10;

对于部分数据,N<=100;

对于部分数据,N<=1000;

对于部分数据,h[i] <= 10;

题解

显然,若存在一个矩形的高度为x,则必然存在一个覆盖的矩形,其上边界高度为x.

所以,若所有h[i]各不相同,就要用N个矩形才能覆盖这个图形.

如果答案小于N,那么必然存在i,j(1<=i<j<=N),使得h[i]=h[j],且对任意k满足i<k<j,有h[k]>=h[i]且h[k]>=h[j](否则该覆盖矩形会超出图形的范围).

单调栈维护严格上升序列即可

#include<iostream>
#include<cstdlib>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<algorithm>
#include<stack>
using namespace std;

int n,ans;
int h[100005];
stack<int> q;

int main()
{
    int i,j;
    scanf("%d",&n);
    ans=n;
    for(i=1;i<=n;i++) scanf("%d",&h[i]);
    q.push(h[1]);
    for(i=2;i<=n;i++)
    {
        while(!q.empty()&&q.top()>=h[i])
        {
            if(q.top()==h[i]) ans--;
            q.pop();
        }
        q.push(h[i]);
    }
    printf("%d",ans);
    return 0;
}

【ZLD NOIP模拟体】序列

【问题描述】

有一个长度为n的非负整数序列,每次操作可以选择相邻的两个数,删去它们,然后在这个位置插入一个数,此次操作的代价定义为,求将这个序列长度变为1的最少代价。

【输入格式】

第一行为一个正整数n,表示序列的长度。

第二行有n个非负整数,表示这个序列。

【输出格式】

一行一个数,表示最少代价。

【输入输出样例】


seq.in


seq.out


3

1 2 3


5

【数据规模】

对于30%的数据,

对于50%的数据,

对于100%的数据,

题解

算法1:我们从左往右读入序列,用一个单调栈,当我们新加进去的数>栈顶时,那么便将栈顶的数拿去跟前一个或者当前的数合并(看哪个比较小),直到不满足这个条件,此时就把当前这个数塞入栈,最后注意一下栈中剩余的元素也是要合并的。

#include<iostream>
#include<cstdlib>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<algorithm>
#include<stack>
using namespace std;
int n;
long long ans;
int a[1000005];
stack<int> q;

int main()
{
    int i,j,temp,in;
    scanf("%d",&n);
    for(i=1;i<=n;i++) scanf("%d",&a[i]);
    q.push(1000000005);
    for(i=1;i<=n;i++)
    {
        if(!q.empty()&&q.top()>a[i])
        {
            q.push(a[i]);
            continue;
        }
        while(!q.empty()&&q.top()<a[i])
        {
            q.pop();
            if(a[i]>q.top()) ans+=(long long)q.top();
            else ans+=(long long)a[i];
        }
        q.push(a[i]);
    }
    if(!q.empty()&&q.size()!=2)
    {
        q.pop();
        while(!q.empty())
        {
            if(q.size()==1) goto hhh;
            ans+=(long long)q.top();
            q.pop();
        }
    }
    hhh:;
    printf("%lld",ans);
    return 0;
}

算法2:

标程做法。我们考虑相邻的两个数x,y,不妨设x>y,则我们先在x右端找一条最长的以y为左端点的区间,使区间内所有的数<=x。

然后呢我们将这些数合并起来,再与x合并,那么这个代价就是x了。

这下子做法就出来了,答案就是

代码略。

时间: 2024-08-25 16:42:16

【NOIP数据结构专项】单调队列单调栈的相关文章

单调队列单调栈

单调队列单调栈 Tags:数据结构 更好阅读体验:https://www.zybuluo.com/xzyxzy/note/1041449 一.概述 单调队列单调栈是很基础的数据结构,常用来优化一些东西比如说优化DP 那么大概意思就是在栈或队列中按照某关键字单调维护信息,从而实现一些功能 其实很久之前接触过单调队列和单调栈,但一直没有刷题,趁这两天被LCT弄晕的时候复习下这些 先看题 二.题单 普及- [x] P1886 滑动窗口 https://www.luogu.org/problemnew/

小Z爱序列(NOIP信(sang)心(bin)赛)From Fall_Dream(粗制单调队列&amp;单调栈的算法解析)

原题: 小Z最擅长解决序列问题啦,什么最长公共上升然后下降然后上升的子序列,小Z都是轻松解决的呢. 但是小Z不擅长出序列问题啊,所以它给了你一道签到题. 给定一个n个数的序列ai,你要求出满足下述条件的点对的数量. 假设点对是(i , j),max(l,r)是[l,r]当中最大的ai的值. 这个点对满足条件当且仅当i+1<j 且 ai < max(i+1,j-1) < aj 为了简单,保证输入的是一个1-n的排列.相信你已经会做了吧? 输入/输出格式 输入数据第一行有一个数字n,然后第二

模板 - 数据结构 - 单调队列/单调栈

一道例题,给定一串数字,求每连续k个数字的最大.最小值. 思路:初始化一个初始长度为k的单调队列,按从左到右加入元素,同时满足这个队列中的元素是递减的(也就是假如某个数被两个距离不超过k的大于他的数夹着,他会被从队尾调出队列).得到最大值. 向右移动一格,假如队首离开范围,出队.往队尾加入元素前,把队尾的所有比它小的元素全部出队. 得到新的最大值. #include<bits/stdc++.h> using namespace std; struct Monotone_Queue{ deque

单调队列&amp;单调栈归纳

单调队列 求长度为M的区间内的最大(小)值 单调队列的基本操作,也就是经典的滑动窗口问题. 求长度为M的区间内最大值和最小值的最大差值 两个单调队列,求出长度为M的区间最大最小值的数组,分别求最大最小值. 求边长为a的正方形内最大值和最小值的最大差值([HAOI2007]理想的正方形) 一个大体的思路是先分别求出以i,j为左上端点的边长为a的矩形中的最大值和最小值.那么该怎么做?,先用单调队列求出一个点右边a个单位的最大值,形成一个新矩阵,再求出新矩阵下边a个单位的最大值.然后最小值再求一边,最

单调队列/单调栈入门详解+题目推荐

以前一直以为这两个是很高级的东西,这段时间用到了才开始学,发现实际上非常简单 下面我们以单调队列为例进行讲解,单调栈自行类比 顾名思义 单调队列这个名字就指明了它的性质--单调性 我们来看一道例题--滑动窗口 题面在此不再赘述,大意就是有一个长度为\(n\)的数列,一个长度为\(k\)的窗口,输出窗口位于每个位置下的下的最大最小值 嗯,题目很好理解,st表或者线段树过的先别说话,我们来看看另一种方法 我们维护一个长度为k的队列,使得队列的开头为答案,那么我们每次只需要输出开头就好了.这个想法很好

单调队列 单调栈

建议不了解STL的读者先了解几个基本的队列的STL.这也是单调队列和单调栈一般都会用到的. 单调队列:建立一个队列,使队列一直具有单调性(满足单调递增或者单调递减),时间复杂度O(N). 那么我们应该如何做到"使队列一直具有单调性"呢? 以单调递增为例,我们O(N)扫描整个序列,每扫描到一个元素: 1 如果该元素大于等于队列末尾元素,则直接入队; 2 而如果该元素小于已有队列的末尾元素,即不满足单调递增,则使队列中的末尾元素出队,直到该元素符合入队条件,然后入队. 如果只到这里,那么我

[hdu1506]单调队列(栈)

题意:http://acm.hdu.edu.cn/showproblem.php?pid=1506看图一目了然.两个方向单调队列维护下. 1 #include <iostream> 2 #include <cstdio> 3 #include <algorithm> 4 #include <cstdlib> 5 #include <cstring> 6 #include <map> 7 #include <queue> 8

算法系列(六)数据结构之表队列和栈

在http://blog.csdn.net/robertcpp/article/details/51559333一文中,我们讲了排序,这一章来介绍一下基本数据结构:表.队列.栈和它们的简单实现 一.表ADT 1.数组实现顺序表 通过对数组操作,来直接对表进行增删查改操作,这种线性表查找某个位置的元素花费的时间为O(1),但是插入删除元素花费的时间为O(n),如果对表的操作更多的是访问操作,那么选择这种实现更为合适. 下面是一个简单实现 package com.algorithm.list; im

数据结构(08)_队列和栈的相互实现

1. 栈的队列的相互实现 思考:栈和队列在实现上非常相似,能否用相互实现? 1.1. StackToQueue 用栈实现队列等价于用"后进先出"的特性实现"先进先出"的特性.实现思路: 准备两个栈用于实现队列:stack_in和stack_out 入队列:当有新元素入队时,将其压入队列stack_in 出队列:当需要出队时:1.stack_out.size() == 0,将stack_in中的数据逐一弹出并压人stack_out(数据移动)2.stack_out.s