hdu 5172 GTY's gay friends

GTY's gay friends

题意:给n个数和m次查询;(1<n,m<1000,000);之后输入n个数值(1 <= ai <= n);问下面m次查询[L,R]中是否存在1~R-L+1的序列;

Sample Input
8 5
2 1 3 4 5 2 3 1
1 3
1 1
2 2
4 8
1 5
 
3 2
1 1 1
1 1
1 2
Sample Output
YES
NO
YES
YES
YES
 
YES
NO
 
分析:问区间是否存在1~R-L+1的排列;注意里面没有一个数相同,并且还都在[1,R-L+1]的区间内;可以等价 每个数前面出现的最大标号一定要小于L(保证了不重复);其次输入的区间和要与结果的和相等;这样就确定了是在这个区间;
利用线段树维护区间pre[](每个点前面出现的最大标号)的最大值;着重理解里面的rt与区间的关系即可;
ps:这道题有更好的算法,使用线段树基本上都 2000+,我的代码2574MS  25448K  AC状态很不满意啊!!
 
#include<iostream>
#include<cstdio>
#include<cstring>
#include<string.h>
#include<algorithm>
#include<map>
#include<queue>
#include<vector>
#include<cmath>
#include<stdlib.h>
#include<time.h>
#include<stack>
#include<set>
using namespace std;
#define rep0(i,l,r) for(int i = (l);i < (r);i++)
#define rep1(i,l,r) for(int i = (l);i <= (r);i++)
#define rep_0(i,r,l) for(int i = (r);i > (l);i--)
#define rep_1(i,r,l) for(int i = (r);i >= (l);i--)
#define MS0(a) memset(a,0,sizeof(a))
#define MS1(a) memset(a,-1,sizeof(a))
#define inf 0x3f3f3f3f
#define lson l, m, rt << 1
#define rson m + 1, r, rt << 1|1
typedef __int64 ll;
template<typename T>
void read1(T &m)
{
    T x=0,f=1;char ch=getchar();
    while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
    while(ch>='0'&&ch<='9'){x=x*10+ch-'0';ch=getchar();}
    m = x*f;
}
template<typename T>
void read2(T &a,T &b){read1(a);read1(b);}
template<typename T>
void read3(T &a,T &b,T &c){read1(a);read1(b);read1(c);}
template<typename T>
void out(T a)
{
    if(a>9) out(a/10);
    putchar(a%10+'0');
}
const int MAXN = 1e6+10;
int id[MAXN],mx[MAXN<<2],pre[MAXN];
ll sum[MAXN];
void pushup(int rt)
{
    mx[rt] = max(mx[rt<<1],mx[rt<<1|1]);
}
void build(int l,int r,int rt)
{
    if(l == r){
        mx[rt] = pre[l];
        return ;
    }
    int m = l + r >> 1;
    build(lson);
    build(rson);
    pushup(rt);
}
int query(int L,int R,int l,int r,int rt)
{
    if(L <= l && r <= R){
        return mx[rt];
    }
    int m = l + r >> 1,ret = 0;
    if(L <= m) ret = max(ret,query(L,R,lson));
    if(R > m) ret = max(ret,query(L,R,rson));
    return ret;
}
int main()
{
    int n,m;
    while(scanf("%d%d",&n,&m) == 2){
        int x;
        fill(id,id+n+1,0);
        rep1(i,1,n){
            sum[i] = 0;
            read1(x);
            sum[i] += sum[i-1] + x;
            pre[i] = id[x];
            id[x] = i;
        }
        build(1,n,1);
        int a,b;
        rep0(i,0,m){
            read2(a,b);
            int len = b-a+1;
            ll sm = (len+1)*len/2;
            if(sm != sum[b] - sum[a-1]) puts("NO");
            else{
                int ret = query(a,b,1,n,1);
                //out(ret);
                puts(ret < a?"YES":"NO");
            }
        }
    }
    return 0;
}
View Code
 
 
原文地址:https://www.cnblogs.com/hxer/p/5191556.html