给一个1到N的排列{Ai},询问是否存在1<=p1<p2<p3<p4<p5<…<pLen<=N (Len>=3),
使得Ap1,Ap2,Ap3,…ApLen是一个等差序列。
标签:span 16px node font str sizeof sam query 数字
题解:显然我们只要让len=3就行。考虑枚举中间数,用桶维护之前已经出现过的数。我们将桶看成一个01串,如果当前数是v[i],考虑桶中v[i]左右两边的串。如果存在v[i]-j和v[i]+j在桶中的存在情况不同,则表明找到了一个等差子序列。即,如果不存在等差子序列,则桶中v[i]的左右两边一定是对称的。用树状数组维护桶的正反hash值即可。
我才不会告诉你这题直接用bitset就能过呢~
#include <cstdio>
#include <cstring>
#include <iostream>
using namespace std;
typedef long long ll;
const ll P=1000000009;
const int maxn=10010;
int n;
int v[maxn];
ll bs[maxn];
struct node
{
ll s[maxn];
inline void updata(int x,ll val)
{
for(int i=x;i<=n;i+=i&-i) s[i]=(s[i]+val)%P;
}
inline ll query(int x)
{
ll ret=0;
for(int i=x;i;i-=i&-i) ret=(ret+s[i])%P;
return ret;
}
}s1,s2;
inline int rd()
{
int ret=0,f=1; char gc=getchar();
while(gc<‘0‘||gc>‘9‘) {if(gc==‘-‘) f=-f; gc=getchar();}
while(gc>=‘0‘&&gc<=‘9‘) ret=ret*10+gc-‘0‘,gc=getchar();
return ret*f;
}
void work()
{
n=rd();
int i,a;
for(i=1;i<=n;i++) v[i]=rd();
memset(s1.s,0,sizeof(s1.s)),memset(s2.s,0,sizeof(s2.s));
for(bs[0]=1,i=1;i<=n;i++) bs[i]=bs[i-1]*131%P;
for(i=1;i<=n;i++)
{
a=min(v[i]-1,n-v[i]);
if((s1.query(v[i]+a)-s1.query(v[i])+P)*bs[n-v[i]+1]%P!=(s2.query(n-v[i]+1+a)-s2.query(n-v[i]+1)+P)*bs[v[i]]%P)
{
printf("Y\n");
return ;
}
s1.updata(v[i],bs[v[i]]),s2.updata(n-v[i]+1,bs[n-v[i]+1]);
}
printf("N\n");
}
int main()
{
int T=rd();
while(T--) work();
return 0;
}//2 3 1 3 2 3 3 2 1
标签:span 16px node font str sizeof sam query 数字
原文地址:http://www.cnblogs.com/CQzhangyu/p/7670397.html