洛谷P4493 [HAOI2018]字串覆盖(后缀自动机+线段树+倍增)

题面

传送门

题解

字符串就硬是要和数据结构结合在一起么……\(loj\)\(rk1\)好像码了\(10k\)的样子……

我们设\(L=r-l+1\)

首先可以发现对于\(T\)串一定是从左到右,能取就取是最优的

我们先用后缀自动机\(+\)线段树合并求出自动机上每一个节点的\(endpos\)集合。如果\(L\)较大的时候,我们考虑二分找到第一个端点,再找下一个……这样在线段树上找的总次数是\({n\over L}\),复杂度为\(O({n\over L}\log n)\)

但是\(L\)较小的时候我们显然不能这样乱蹦,那么我们对于每一个位置,维护一个\(f[L][i]\),表示一个长度为\(L\)的以\(i\)位置结尾的子串,下一个与它相同的位置在哪儿。这个对于所有\(L\)相同的询问显然可以统一处理。那么对于一个询问,只要知道它往后跳了几步,跳的这几步的位置的\(i\)之和就可以了,倍增即可

话说\(unordered\_map\)的头文件在\(c++11\)里居然会\(CE\)么……

//minamoto
#include<bits/stdc++.h>
#include<tr1/unordered_map>
#define R register
#define ll long long
#define ull unsigned long long
#define fp(i,a,b) for(R int i=(a),I=(b)+1;i<I;++i)
#define fd(i,a,b) for(R int i=(a),I=(b)-1;i>I;--i)
#define go(u) for(int i=head[u],v=e[i].v;i;i=e[i].nx,v=e[i].v)
using namespace std;
using namespace std::tr1;
char buf[1<<21],*p1=buf,*p2=buf;
inline char getc(){return p1==p2&&(p2=(p1=buf)+fread(buf,1,1<<21,stdin),p1==p2)?EOF:*p1++;}
int read(){
    R int res,f=1;R char ch;
    while((ch=getc())>'9'||ch<'0')(ch=='-')&&(f=-1);
    for(res=ch-'0';(ch=getc())>='0'&&ch<='9';res=res*10+ch-'0');
    return res*f;
}
int read(char *s){
    R int len=0;R char ch;while(((ch=getc())>'z'||ch<'a'));
    for(s[++len]=ch;(ch=getc())>='a'&&ch<='z';s[++len]=ch);
    return s[len+1]='\0',len;
}
char sr[1<<21],z[20];int C=-1,Z=0;
inline void Ot(){fwrite(sr,1,C+1,stdout),C=-1;}
void print(R ll x){
    if(C>1<<20)Ot();if(x<0)sr[++C]='-',x=-x;
    while(z[++Z]=x%10+48,x/=10);
    while(sr[++C]=z[Z],--Z);sr[++C]='\n';
}
const int N=2e5+5,M=N*20+5,Base=131;
struct node{
    int s,t,l,r,id,L;
    inline void init(R int Id){s=read(),t=read(),l=read(),r=read(),id=Id,L=r-l+1;}
    inline bool operator <(const node &b)const{return L<b.L;}
}q[N];
char s[N],t[N];
int ch[N][26],fa[N],l[N],rt[N],f[N][21],F[N][21],G[N][21],lc[M],rc[M],sz[M];ll ans[N];
int c[N],a[N],id[N],mx[N],poi[N],Log[N],cl[N],zz[N];ull bin[N],h[N];
int las=1,cnt=1,tot,n,k,T;unordered_map<ull,int>mp;
inline ull gh(R int l,R int r){return h[r]-h[l-1]*bin[r-l+1];}
void ins(int c){
    int p=las,np=las=++cnt;l[np]=l[p]+1;
    for(;p&&!ch[p][c];p=fa[p])ch[p][c]=np;
    if(!p)fa[np]=1;
    else{
        int q=ch[p][c];
        if(l[q]==l[p]+1)fa[np]=q;
        else{
            int nq=++cnt;l[nq]=l[p]+1;
            memcpy(ch[nq],ch[q],4*26);
            fa[nq]=fa[q],fa[q]=fa[np]=nq;
            for(;p&&ch[p][c]==q;p=fa[p])ch[p][c]=nq;
        }
    }
}
void upd(int &p,int l,int r,int x){
    if(!p)p=++tot;++sz[p];if(l==r)return;
    int mid=(l+r)>>1;
    x<=mid?upd(lc[p],l,mid,x):upd(rc[p],mid+1,r,x);
}
int merge(int x,int y){
    if(!x||!y)return x|y;
    int p=++tot;
    lc[p]=merge(lc[x],lc[y]),rc[p]=merge(rc[x],rc[y]);
    sz[p]=sz[lc[p]]+sz[rc[p]];return p;
}
int find(int p,int l,int r){
    if(l==r)return l;
    int mid=(l+r)>>1;
    return sz[lc[p]]?find(lc[p],l,mid):find(rc[p],mid+1,r);
}
int query(int p,int l,int r,int ql,int qr){
    if(!sz[p])return -1;if(ql<=l&&qr>=r)return find(p,l,r);
    int mid=(l+r)>>1,res=-1;
    if(ql<=mid&&(res=query(lc[p],l,mid,ql,qr))!=-1)return res;
    if(qr>mid&&(res=query(rc[p],mid+1,r,ql,qr))!=-1)return res;
    return -1;
}
void solve_min(int p,int l,int r,int L,int id){
    ll c=0,d=0;int x=query(rt[p],1,n,l,r);
    if(x==-1)return ans[id]=0,void();
    fd(i,zz[x],0)if(F[x][i]&&F[x][i]<=r)c+=(1<<i),d+=G[x][i],x=F[x][i];
    ++c,d+=x,ans[id]=c*k-d+(L-1)*c;
}
void solve_max(int p,int l,int r,int L,int id){
    ll c=0,d=0;int x;
    while(l<=r){
        x=query(rt[p],1,n,l,r);
        if(x==-1)break;
        ++c,d+=x,l=x+L;
    }
    ans[id]=c*k-d+(L-1)*c;
}
void init(){
    n=read(),k=read(),read(s),read(t);
    fp(i,1,n)ins(s[i]-'a'),upd(rt[las],1,n,i);
    fp(i,1,cnt)f[i][0]=fa[i],++c[l[i]];
    for(R int j=1;(1<<j)<=cnt;++j)fp(i,1,cnt)f[i][j]=f[f[i][j-1]][j-1];
    fp(i,1,cnt)c[i]+=c[i-1];
    fd(i,cnt,1)id[c[l[i]]--]=i;
    for(R int i=cnt,p=id[i];i>1;--i,p=id[i])
        rt[fa[p]]=merge(rt[fa[p]],rt[p]);
    for(R int i=1,p=1,now=0;i<=n;++i){
        R int c=t[i]-'a';
        if(ch[p][c])p=ch[p][c],++now;
        else{
            for(;p&&!ch[p][c];p=fa[p]);
            p?(now=l[p]+1,p=ch[p][c]):(p=1,now=0);
        }
        mx[i]=now,poi[i]=p;
    }
    bin[0]=1,Log[0]=-1;
    fp(i,1,n)bin[i]=bin[i-1]*Base,h[i]=h[i-1]*Base+s[i]-'a'+1,Log[i]=Log[i>>1]+1;
}
void solve(){
    T=read();fp(i,1,T)q[i].init(i);
    sort(q+1,q+1+T);
    int pos=1,s,t,l,r,id,p,L;
    for(R int L=1;L<=50;++L)if(q[pos].L==L){
        fd(i,n,L){
            R ull tmp=gh(i-L+1,i);
            if(mp.count(tmp)){
                R int x=mp[tmp],sz=Log[cl[x]+1];
                zz[i]=sz,F[i][0]=x,G[i][0]=i,cl[i]=cl[x]+1;
                fp(j,1,sz){
                    x=F[i][j-1];
                    F[i][j]=F[x][j-1],G[i][j]=G[i][j-1]+G[x][j-1];
                }
            }else cl[i]=0;
            if(i+L-1<=n)mp[gh(i,i+L-1)]=i+L-1;
        }
        unordered_map<ull,int>().swap(mp);
        for(;pos<=T&&q[pos].L==L;++pos){
            s=q[pos].s,t=q[pos].t,l=q[pos].l,r=q[pos].r,id=q[pos].id;
            if(mx[r]<L){ans[id]=0;continue;}
            p=poi[r];
            fd(j,Log[::l[p]],0)if(::l[f[p][j]]>=L)p=f[p][j];
            solve_min(p,s+L-1,t,L,id);
        }
        fd(i,n,L)fp(j,0,zz[i])F[i][j]=G[i][j]=0;
    }
    for(;pos<=T;++pos){
        s=q[pos].s,t=q[pos].t,l=q[pos].l,r=q[pos].r,id=q[pos].id,L=r-l+1;
        if(mx[r]<L){ans[id]=0;continue;}
        p=poi[r];
        fd(j,Log[::l[p]],0)if(::l[f[p][j]]>=L)p=f[p][j];
        solve_max(p,s+L-1,t,L,id);
    }
    fp(i,1,T)print(ans[i]);
}
int main(){
//  freopen("testdata.in","r",stdin);
    init(),solve();
    return Ot(),0;
}

转载于:https://www.cnblogs.com/bztMinamoto/p/10522427.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值