BZOJ 2124: 等差子序列

Sol

线段树+Hash.

首先暴力 等差子序列至少3项就可以了,就枚举中项,枚举公差就可以了,只需要一个数在中项前出现,另一个数在中项前没出现过就可以了.复杂度 \(O(n^2)\)

然后我想了一个虽然复杂度没变(因为我不会设计这个数据结构...) 但是好像有点用的算法,就是枚举中项,考虑从一个中项转移到另一个中项,那就是 \(\pm \Delta\) 就可以了...如果能够用数据结构维护这个操作,那就灰常好了.变换中项也就是变换折叠的位置吧.

标算呢...就是用线段树维护Hash,一个数字出现过,那就为1,没出现过就为0,维护一个正反序的01序列,如果一个数的 小于它正序01序列 和 大于它的反序01序列 异或值不为0那么就存在满足条件的等差序列.

01串太长了就可以用Hash来维护了...

PS:100000007不是质数...没开Long Long 直接见祖宗.

PS:我常数好大啊QAQ

Code

/**************************************************************
    Problem: 2124
    User: BeiYu
    Language: C++
    Result: Accepted
    Time:2348 ms
    Memory:8720 kb
****************************************************************/
 
#include<cstdio>
#include<cstring>
#include<iostream>
using namespace std;
 
typedef long long LL;
const int N = 100005;
const LL p = 100000007;
#define debug(a) cout<<#a<<"="<<a<<" "
#define mid ((l+r)>>1)
#define lc (o<<1)
#define rc (o<<1|1)
 
int T,n;
int a[N];
LL h1[N<<2],h2[N<<2],pow[N];
 
inline int in(int x=0,char ch=getchar()){ while(ch>'9'||ch<'0') ch=getchar();
    while(ch>='0'&&ch<='9') x=(x<<3)+(x<<1)+ch-'0',ch=getchar();return x; }
 
inline void Update(int x,int o,int l,int r){
    if(l==r){ h1[o]=h2[o]=1;return; }
    if(x<=mid) Update(x,lc,l,mid);
    else Update(x,rc,mid+1,r);
    h1[o]=(h1[lc]+h1[rc]*pow[mid-l+1])%p;
    h2[o]=(h2[lc]*pow[r-mid]+h2[rc])%p;
}
inline LL Query1(int L,int R,int o,int l,int r){
    if(L>R) return -1;
//  debug(L),debug(R),debug(l),debug(r)<<endl;
    if(L==l&&r==R) return h1[o];
    if(L<=mid&&R>mid) return (Query1(L,mid,lc,l,mid)+Query1(mid+1,R,rc,mid+1,r)*pow[mid-L+1])%p;
    if(L<=mid) return Query1(L,R,lc,l,mid)%p;
    else return Query1(L,R,rc,mid+1,r)%p;
//  debug(res)<<endl;
//  return res;
}
inline LL Query2(int L,int R,int o,int l,int r){
    if(L>R) return -1;
    if(L==l&&r==R) return h2[o];
    if(L<=mid&&R>mid) return (Query2(L,mid,lc,l,mid)*pow[R-mid]+Query2(mid+1,R,rc,mid+1,r))%p;
    if(L<=mid) return Query2(L,R,lc,l,mid)%p;
    else return Query2(L,R,rc,mid+1,r)%p;
//  return res;
}
int main(){
    pow[0]=1;for(int i=1;i<N;i++) pow[i]=(pow[i-1]<<1)%p;
    for(T=in();T--;){
        n=in();for(int i=1;i<=n;i++) a[i]=in();
        int f=0;
        memset(h1,0,sizeof(h1)),memset(h2,0,sizeof(h2));
        for(int i=1;i<=n;i++){
            int l=min(n-a[i],a[i]-1);
            int u=Query2(a[i]-l,a[i]-1,1,1,n);
            int v=Query1(a[i]+1,a[i]+l,1,1,n);
//          cout<<"*********"<<endl;
//          debug(i)<<endl;
//          debug(l),debug(a[i])<<endl;
//          debug(a[i]-l),debug(a[i]-1)<<endl;
//          debug(a[i]+1),debug(a[i]+l)<<endl;
//          cout<<u<<" "<<v<<endl; 
            if(u!=v){ f=1;break; }
            Update(a[i],1,1,n);
        }if(f) puts("Y");else puts("N");
    }
    return 0;
}

  

posted @ 2016-09-10 14:10  北北北北屿  阅读(245)  评论(0编辑  收藏  举报