RQNOJ460 诺诺的队列

Posted liguangsunls

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了RQNOJ460 诺诺的队列相关的知识,希望对你有一定的参考价值。

【题目大意】
求全部数对(i,j)满足随意a[k]<=a[i]且a[k]<=a[j]。
形象地说,就是有一群人站成一列。每一个人有一定的身高,然后问有多少对人能够互相看得到。
把数对(i,j)简单地称之为看得到的数对。

【解析】单调栈
先借用一下曾经做的题:[Vijos]1926 紫色的手链。求随意区间最大值异或次大值的最大值。
回想一下单调栈,就是存储从高到低递减的单调数据的栈。


借用曾经的做法,求出来的东西相对于全部看得到的数对,对于全部a[i]相等的看得见的数对,仅仅算了一次。

于是事实上每次高过别人的时候操作仅仅要加上s(a[i]),第一次比别人矮的时候加上1而不是s(a[i])。


把栈内的东西给扩充,不仅存元素。还存个数,这样就攻克了。

单调栈代码:

#include <cstdio>
#include <cstring>
#include <cstdlib>
using namespace std;

const int N=500001;

struct Stack
{
	int w,c;
}stk[N];
int size;
int n,w[N],cnt;

inline int read(void)
{
	int s=0,f=1; char c=getchar();
	for (;c<'0'||c>'9';c=getchar()) if (c=='-') f=-1;
	for (;'0'<=c&&c<='9';c=getchar()) s=(s<<1)+(s<<3)+c-'0';
	return s*f;
}

int main(void)
{	
	n=read();
	for (int i=1;i<=n;i++) w[i]=read();
	
	for (int i=1;i<=n;i++)
	{
		for (;size&&stk[size].w<w[i];size--)
		{
			cnt+=stk[size].c;
			stk[size].w=stk[size].c=0;
		}
		if (size&&stk[size].w==w[i])
		{
			cnt+=stk[size].c;
			stk[size].c++;
			if (size-1) cnt++;
		}
		else
		{
			if (size) cnt++;
			stk[++size].w=w[i];
			stk[size].c=1;
		}
	}
	printf("%d\n",cnt);
	
	return 0;
}


以上是关于RQNOJ460 诺诺的队列的主要内容,如果未能解决你的问题,请参考以下文章

# Java 常用代码片段

谈谈诺诺商城的交互设计

RQNOJ PID192 梦幻大PK [2017年6月计划 二分图02]

PID736(rqnoj)

RQNOJ 429 词链:单调栈

RQNOJ 169 最小乘车费用:水dp