hdu 5172(RMQ+前缀和)

题意:判断一个区间[l,r]的数是否是1到(r-l+1)。

解题思路:首先判断该区间内的和是否是n*(1+n)/2,这里可以用前缀和判断。

接下来是要判断该区间内无重复的数:

pos[i]表示第i个数左边与它相同的且最靠近它的位置。如果寻找的[l,r]区间内最大的pos[i]大于了l,说明区间有重复的数。这里可以用rmq维护,但rmq超内存。正解是线段树。


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

const int maxn = 1000005;
int n,m,sum[maxn],tmp[maxn];
int pos[maxn],dp[maxn][20];

void init()
{
	for(int i = 1; i <= n; i++)
		dp[i][0] = pos[i];
	for(int j = 1; j <= 20; j++)
		for(int i = 1; i + (1 << j) - 1 <= n; j++)
			dp[i][j] = max(dp[i][j-1],dp[i+(1<<(j-1))][j-1]);
}

int rmq(int l,int r)
{
	int k = (int)(log(r - l + 1.0) / log(2.0));
	return max(dp[l][k],dp[r-(1<<k)+1][k]);
}

int main()
{
	int l,r;
	while(scanf("%d%d",&n,&m)!=EOF)
	{
		memset(tmp,0,sizeof(tmp));
		for(int i = 1; i <= n; i++)
		{
			scanf("%d",&sum[i]);
			pos[i] = tmp[sum[i]];
			tmp[sum[i]] = i;
		}
		for(int i = 2; i <= n; i++)
			sum[i] += sum[i-1];
		init();
		while(m--)
		{
			scanf("%d%d",&l,&r);
			if((1 + (r - l + 1)) * (r - l + 1) / 2 != sum[r] - sum[l-1])
			{
				printf("NO\n");
				continue;
			}
			if(rmq(l,r) < l)
				printf("YES\n");
			else printf("NO\n");
		}
	}
	return 0;
}


你可能感兴趣的:(数据结构)