题意:
给出n个数,形成一个长度为n的序列(可以看做[1,n]的区间)
有m组询问,对于每组询问给出一个x和一个y,然后给出一个长度为k的序列s。如果在区间[x,y]上,存在至少一个序列和序列s完全匹配,那么输出No,否则输出Yes。
**注意,匹配成功了输出No,失败时才会输出Yes
思路:嘛…就是把每k个数字hash一下存进可持久化线段树的对应位置,这样得到了n-k+1个线段树的根节点,分别为[k,n]。每次查询第x+k-2到第y个线段树对应位置的size[]是否发生变化(差非0),如果发生了变化就说明这段区间存在size[y]-size[x+k-2]个相匹配的序列
hash我实在不会QwQ借黄学长的hash方式过了
#include<iostream>
#include<stdio.h>
#define ull unsigned long long
#define inf 18446744073709551615UL
using namespace std;
const int maxn=200005;
int n,m,k,x,y,a,tot,val[maxn],s[maxn*40],son[maxn*40][2];
ull rt[maxn],v[maxn],M=1,b;
void addtree(ull l,ull r,int rl,int rn,ull value)
{
s[rn]=s[rl]+1;
if(l==r)
return;
ull mid=l/2+r/2;
mid+=(l&1)&&(r&1);
if(value>mid)son[rn][1]=++tot,son[rn][0]=son[rl][0],addtree(mid+1,r,son[rl][1],son[rn][1],value);
else son[rn][0]=++tot,son[rn][1]=son[rl][1],addtree(l,mid,son[rl][0],son[rn][0],value);
}
void insert(int a,int b,ull c)
{
rt[b]=++tot;
addtree(1,inf,rt[a],rt[b],c);
}
int query(ull l,ull r,int rl,int rn,ull value)
{
if(l==r)
return s[rn]-s[rl];
ull mid=l/2+r/2;
mid+=(l&1)&&(r&1);
if(value>mid)return query(mid+1,r,son[rl][1],son[rn][1],value);
else return query(l,mid,son[rl][0],son[rn][0],value);
}
int main(void)
{
scanf("%d%d%d",&n,&m,&k);
for(int i=1;i<=n;i++)
{
scanf("%d",&val[i]);
val[i]++;
}
for(int i=1;i<=n;i++)
v[i]=v[i-1]*107,v[i]+=(ull)val[i];
for(int i=1;i<=k;i++)
M*=107;
for(int i=k;i<=n;i++)
insert(i-1,i,v[i]-v[i-k]*M);
for(int i=1;i<=m;i++)
{
scanf("%d%d",&x,&y);
b=0;
for(int j=1;j<=k;j++)
{
scanf("%d",&a);
b*=107;
b+=(ull)(a+1);
}
if(query(1,inf,rt[x+k-2],rt[y],b))printf("No\n");
else printf("Yes\n");
}
return 0;
}