hdu 5172 (RMQ + 접두사 와)
1505 단어 데이터 구조
문제 풀이 방향: 먼저 이 구간 안의 것 과 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
");
continue;
}
if(rmq(l,r) < l)
printf("YES
");
else printf("NO
");
}
}
return 0;
}
이 내용에 흥미가 있습니까?
현재 기사가 여러분의 문제를 해결하지 못하는 경우 AI 엔진은 머신러닝 분석(스마트 모델이 방금 만들어져 부정확한 경우가 있을 수 있음)을 통해 가장 유사한 기사를 추천합니다:
정수 반전Udemy 에서 공부 한 것을 중얼거린다 Chapter3【Integer Reversal】 (예) 문자열로 숫자를 반전 (toString, split, reverse, join) 인수의 수치 (n)가 0보다 위 또는 ...
텍스트를 자유롭게 공유하거나 복사할 수 있습니다.하지만 이 문서의 URL은 참조 URL로 남겨 두십시오.
CC BY-SA 2.5, CC BY-SA 3.0 및 CC BY-SA 4.0에 따라 라이센스가 부여됩니다.