BZOJ 3207 花神的嘲讽计划Ⅰ 题解&代码

题意:
给出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;
}

你可能感兴趣的:(题解,C++,线段树,主席树)