P4770-[NOI2018]你的名字【SAM,线段树合并】

news/2024/11/24 18:47:44/

正题

题目链接:https://www.luogu.com.cn/problem/P4770


题目大意

给出一个长度为 n n n的字符串 S S S q q q次询问给出一个串 T T T和一个区间 [ L , R ] [L,R] [L,R],求 T T T有多少个本质不同的子串不是 S L ∼ R S_{L\sim R} SLR的子串。

1 ≤ n ≤ 5 × 1 0 5 , 1 ≤ Q ≤ 1 0 5 , ∑ ∣ T ∣ ≤ 1 0 6 1\leq n\leq 5\times 10^5,1\leq Q\le 10^5,\sum|T|\leq 10^6 1n5×105,1Q105,T106


解题思路

因为给了很多 L = 1 , R = n L=1,R=n L=1,R=n的部分分所以应该是提示我们先从这个方面考虑。
这个部分比较简单,考虑改为求有多少个本质不同的子串在 S L ∼ R S_{L\sim R} SLR中出现过,因为是本质不同的子串,我们可以先建一个 S S S S A M SAM SAM和一个 T T T S A M SAM SAM

然后把 T T T串拿到 S S S S A M SAM SAM上面跑,然后每次跑出来的一个匹配长度记为 l e n len len。对于 T T T S A M SAM SAM上的每一个节点我们记录一个 p o s pos pos表示这个节点属于的长度位置,然后跑到这个位置的 l e n len len就是能够匹配的长度了,记为 a n s ans ans。然后答案就是 m a x { l e n i − m a x { a n s p o s i , l e n f a i } , 0 } max\{len_i-max\{ans_{pos_i},len_{fa_i}\},0\} max{lenimax{ansposi,lenfai},0}(防匹配长度超出 [ l e n f a i , l e n i ] [len_{fa_i},len_i] [lenfai,leni]的范围)

这样一次的时间复杂度就是 O ( ∣ T ∣ ) O(|T|) O(T)的了。

然后考虑带区间的怎么做,我们需要保证我们在 S S S S A M SAM SAM上面跳的时候需要保证这些节点都是属于 S L ∼ R S_{L\sim R} SLR的自动机上的,而且要保证我们提取出来的 l e n len len也是在那个上面的。

其实如果这个节点的 e n d p o s endpos endpos类里面有 L ∼ R L\sim R LR的信息就好了,这个用线段树合并维护一下 e n d p o s endpos endpos类的信息就可以了。

时间复杂度 O ( ( n + ∑ ∣ T ∣ ) log ⁡ n ) O((n+\sum |T|)\log n) O((n+T)logn)


code

#include<cstdio>
#include<cstring>
#include<algorithm>
#define ll long long
using namespace std;
const int N=1e6+10;
int n,q,ql,qr,p[N],c[N],rt[N],ans[N],pos[N];
char s[N];
struct SegTree{int cnt,w[N<<5],ls[N<<5],rs[N<<5];int Change(int x,int L,int R,int pos){int p=++cnt;w[p]=max(w[x],pos);if(L==R)return p;int mid=(L+R)>>1;if(pos<=mid)ls[p]=Change(ls[x],L,mid,pos),rs[p]=rs[x];else rs[p]=Change(rs[x],mid+1,R,pos),ls[p]=ls[x];return p;}int Ask(int x,int L,int R,int l,int r){if(!x)return 0;if(L==l&&R==r)return w[x];int mid=(L+R)>>1;if(r<=mid)return Ask(ls[x],L,mid,l,r);if(l>mid)return Ask(rs[x],mid+1,R,l,r);return max(Ask(ls[x],L,mid,l,mid),Ask(rs[x],mid+1,R,mid+1,r));}int Merge(int x,int y){if(!x||!y)return x+y;int p=++cnt;w[p]=max(w[x],w[y]);ls[p]=Merge(ls[x],ls[y]);rs[p]=Merge(rs[x],rs[y]);return p;}
}R;
struct SAM{int cnt,last,ch[N][26],len[N],fa[N];void init(){memset(ch[1],0,sizeof(ch[1]));last=cnt=1;}int Insert(int c){int p=last,np=last=++cnt;len[np]=len[p]+1;memset(ch[np],0,sizeof(ch[np]));for(;p&&!ch[p][c];p=fa[p])ch[p][c]=np;if(!p)fa[np]=1;else{int q=ch[p][c];if(len[p]+1==len[q])fa[np]=q;else{int nq=++cnt;len[nq]=len[p]+1;memcpy(ch[nq],ch[q],sizeof(ch[nq]));fa[nq]=fa[q];fa[q]=fa[np]=nq;pos[nq]=pos[q];for(;p&&ch[p][c]==q;p=fa[p])ch[p][c]=nq;}}last=np;return np;}void Build(){for(int i=1;i<=cnt;i++)c[len[i]]++;for(int i=1;i<=n;i++)c[i]+=c[i-1];for(int i=1;i<=cnt;i++)p[c[len[i]]--]=i;for(int i=cnt;i>=1;i--){int x=p[i];rt[fa[x]]=R.Merge(rt[fa[x]],rt[x]);}return;}void next(int &x,int &l,int c){while(x){if(ch[x][c]){int maxl=R.Ask(rt[ch[x][c]],1,n,1,qr)-ql+1;if(len[fa[x]]<maxl){l=min(l+1,maxl);x=ch[x][c];return;}}x=fa[x];l=len[x];}l=0;x=1;return;}ll calc(){ll prt=0;for(int i=2;i<=cnt;i++)prt+=max(len[i]-max(ans[pos[i]],len[fa[i]]),0);return prt;}
}S,T;
void work(char *s){int m=strlen(s+1);T.init();for(int i=1;i<=m;i++){int x=T.Insert(s[i]-'a');pos[x]=i;}int x=1,l=0;for(int i=1;i<=m;i++){int c=s[i]-'a';S.next(x,l,c);ans[i]=l;}printf("%lld\n",T.calc());
}
signed main()
{scanf("%s",s+1);n=strlen(s+1);S.init();for(int i=1;i<=n;i++){int x=S.Insert(s[i]-'a');rt[x]=R.Change(rt[x],1,n,i);}S.Build();scanf("%d",&q);while(q--){scanf("%s",s+1);scanf("%d%d",&ql,&qr);work(s);}return 0;
}

http://www.ppmy.cn/news/147402.html

相关文章

corei7 64 poky linux,Core i7-4770K Linux之旅:有喜有忧

Core i7-4770K Linux之旅&#xff1a;有喜有忧 出处&#xff1a;快科技 2013-06-09 11:37:15 作者&#xff1a;上方文Q 编辑&#xff1a;上方文Q[爆料] 收藏文章 Haswell的评测多如牛毛&#xff0c;但都是在Windows下进行的&#xff0c;Linux用户肯定看不下去了。Phoronix又…

酷睿i7cpu适合的linux,CPU性能篇 - Core i7-4770K Linux之旅:有喜有忧_Linux新闻_Linux公社-Linux系统门户网站...

CPU性能篇—— Rodinia是学术界经常使用的科学测试工具。OpenMP LavaMD负载中&#xff0c;4770K相比3770K快了12&#xff05;&#xff0c;8350表现也可以。 OpenMP Leukocyte负载里&#xff0c;4770K对比3770K的优势依然有10&#xff05;&#xff0c;但是8350大亮了&#xff0c…

HDU 4770

这道题利用DFS进行枚举就可以了。 由于图中的点很多&#xff0c;但是vulnerable room 很少&#xff0c;所以要把这些点保存起来。此外&#xff0c;由于回溯的时候判断哪些room任然被灯照射有些麻烦&#xff0c;所以用状态压缩的方式表示一个vulnerable room 被照射的状态&#…

BZOJ4770 图样

牛逼的DP&#xff0c;一股TC气息 求期望的话可以先求出所有情况的和再除以情况数 考虑 f[i][j] 表示 i 个点每个点的权值都是j位的二进制数&#xff0c;最小生成树的和 一定是最高位为0的和为1的分别的形成生成树&#xff0c;然后两部分之间连一条边 那么枚举 i 个点里有多…

群晖服务器性能测试,原创首发!群晖J3455 G4560 I7 4770HQ功耗性能测试!

本帖最后由 -我傻也要中二 于 2019-4-28 16:58 编辑 写在最前&#xff1a;本人也是新手入得群晖&#xff0c;一开始直接买的J3455&#xff0c;群晖的强大的功能&#xff0c;嘿嘿舒服。不过因为我本身是有一台windows服务器用来搭建ERP的&#xff0c;考虑到群晖可以运行虚拟机就…

4770: 栈操作

定义一个栈&#xff0c;可以对栈进行“压入堆栈”、“弹出栈顶元素”、“清空堆栈”、“获取栈顶元素”等操作。键盘输入一些命令&#xff0c;可以执行上述操作。本题中&#xff0c;栈元素均为整数。栈的最大元素个数为1000。 输入 输入各个命令&#xff0c;它们对应的格式如…

碧砚适合佳能328 4452 ICD520 4472 4450 硒鼓4700一体机墨盒4770

转载于:https://www.cnblogs.com/zengkefu/p/6036577.html

【JavaSE】方法的使用--05

目录 一、方法的概念与使用 1.1 什么是方法 1.2 方法的定义 1.3 方法调用的执行过程 1.4 实参和形参的关系&#xff08;重要&#xff09; 1.5 有无返回值的方法 二、方法的重载 2.1 方法重载的概念 2.2 方法重载的要求 2.3 方法签名 前言&#xff1a; 之前很久没写这…