1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89
| #include<cstdio> #include<cstring> #include<iostream> using namespace std; struct SAM{ int ch[10],len,fa; }t1[200001]; struct tree{ int size,ch[2]; }t2[200001*30]; char ch[100001],s[100001]; int root[200001],tot_t,tot_s=1,last=1,n,m,pos[200001]; int o[100001],a[200001],ans; int update(int rt,int l,int r,int x){ int u=++tot_t; t2[u]=t2[rt]; t2[u].size++; if (l==r) return u; int mid=(l+r)>>1; if (x<=mid) t2[u].ch[0]=update(t2[rt].ch[0],l,mid,x); else t2[u].ch[1]=update(t2[rt].ch[1],mid+1,r,x); return u; } int merge(int c,int d,int l,int r){ if (!c||!d) return c|d; int u=++tot_t; t2[u].size=t2[c].size+t2[d].size; if (l==r) return u; int mid=(l+r)>>1; t2[u].ch[0]=merge(t2[c].ch[0],t2[d].ch[0],l,mid); t2[u].ch[1]=merge(t2[c].ch[1],t2[d].ch[1],mid+1,r); return u; } int query(int rt,int l,int r,int x){ if (r<=x) return t2[rt].size; int mid=(l+r)>>1,num=query(t2[rt].ch[0],l,mid,x); if (x>mid) num+=query(t2[rt].ch[1],mid+1,r,x); return num; } inline void extend(int x){ int np=++tot_s,p=last; last=np; t1[np].len=t1[p].len+1; while (p&&!t1[p].ch[x]) t1[p].ch[x]=np,p=t1[p].fa; if (!p) t1[np].fa=1; else{ int q=t1[p].ch[x]; if (t1[q].len==t1[p].len+1) t1[np].fa=q; else{ int cp=++tot_s; t1[cp]=t1[q]; t1[cp].len=t1[p].len+1; t1[q].fa=t1[np].fa=cp; while (p&&t1[p].ch[x]==q) t1[p].ch[x]=cp,p=t1[p].fa; } } } int main(){ scanf("%d",&n); scanf("%s",ch+1); memset(pos,10,sizeof(pos)); pos[0]=0; for (int i=1; i<=n; i++){ extend(ch[i]-'0'); pos[last]=i; root[last]=update(root[last],1,n,i); } for (int i=1; i<=tot_s; i++) o[t1[i].len]++; for (int i=1; i<=n; i++) o[i]+=o[i-1]; for (int i=1; i<=tot_s; i++) a[o[t1[i].len]--]=i; int u,ed,len; for (int i=tot_s; i>1; i--){ u=a[i]; pos[t1[u].fa]=min(pos[t1[u].fa],pos[u]); root[t1[u].fa]=merge(root[t1[u].fa],root[u],1,n); } scanf("%d",&m); while (m--){ scanf("%s",s+1); len=strlen(s+1); u=1; for (int i=1; i<=len&&u; i++) u=t1[u].ch[s[i]-'0']; ed=pos[u]; u=1; ans=0; for (int i=1; i<=len; i++){ u=t1[u].ch[s[i]-'0']; if (!u) break; if (!ed) ans+=query(root[u],1,n,n); else ans+=query(root[u],1,n,ed+i-len); } if (!ed) ans+=n; else ans+=ed-len; printf("%d\n",ans); } return 0; }
|