洛谷P3804 - 【模板】后缀自动机

Portal

Description

模板题啦~
推荐WJMZBMR在WC2012上的讲义《后缀自动机》

Code

//【模板】后缀自动机
#include <cstdio>
int max(int x,int y) {return x>y?x:y;}
int const N=2e6+10;
int n; char s[N];
int rt,ndCnt,last;
int fa[N],ch[N][26],len[N],siz[N];
void ins(int x)
{
    int p=last,np=++ndCnt;
    last=np,len[np]=len[p]+1; siz[np]=1;
    for(p;!ch[p][x]&&p;p=fa[p]) ch[p][x]=np;
    if(!p) {fa[np]=rt; return;}
    int q=ch[p][x];
    if(len[p]+1==len[q]) {fa[np]=q; return;}
    int nq=++ndCnt; len[nq]=len[p]+1;
    for(int i=0;i<26;i++) ch[nq][i]=ch[q][i];
    fa[nq]=fa[q]; fa[q]=fa[np]=nq;
    for(p;ch[p][x]==q;p=fa[p]) ch[p][x]=nq;
}
int cnt[N],t[N];
int query()
{
    for(int i=1;i<=ndCnt;i++) cnt[len[i]]++;
    for(int i=1;i<=ndCnt;i++) cnt[i]+=cnt[i-1];
    for(int i=1;i<=ndCnt;i++) t[cnt[len[i]]--]=i;
    int res=0;
    for(int i=ndCnt;i>=1;i--)
    {
        int p=t[i];
        siz[fa[p]]+=siz[p];
        if(siz[p]>1) res=max(res,1LL*siz[p]*len[p]);
    }
    return res;
}
int main()
{
    scanf("%s",s+1);
    last=1,rt=++ndCnt;
    for(n=1;s[n];n++) ins(s[n]-'a');
    printf("%lld\n",query());
    return 0;
}

原文地址:https://www.cnblogs.com/VisJiao/p/LgP3804.html

时间: 2024-11-13 13:29:53

洛谷P3804 - 【模板】后缀自动机的相关文章

洛谷 P3804 后缀自动机

题目描述 给定一个只包含小写字母的字符串SS , 请你求出 SS 的所有出现次数不为 11 的子串的出现次数乘上该子串长度的最大值. 输入输出格式 输入格式: 一行一个仅包含小写字母的字符串SS 输出格式: 一个整数,为 所求答案 输入输出样例 输入样例#1: abab 输出样例#1: 4 说明 对于10\%10% 的数据,|S|<=1000∣S∣<=1000 对于100\%100% 的数据,|S|<=10^6∣S∣<=106 今天上午恶补了一下后缀自动机,发现好像没有想象中的那么

AC自动机(附洛谷P3769模板题)

首先,介绍一下AC自动机(Aho-Corasick automaton),是一种在一个文本串中寻找每一个已给出的模式串的高效算法. 在学习AC自动机之前,你需要先学习Trie树和KMP算法,因为AC自动机正式利用并结合了两者的思想. 说到实际的不同,其实AC自动机只是在Trie树上引入了一个类似KMP中next数组的东西叫做Fail指针. 对于每一个节点,Fail指针指向该节点所代表的字符串中,次长的.在Trie树中存在的后缀(因为最长的在Trie树种存在的后缀就是其本身)所代表的节点. 举例:

【C++】最近公共祖先LCA(Tarjan离线算法)&amp;&amp; 洛谷P3379LCA模板

1.前言 首先我们介绍的算法是LCA问题中的离线算法-Tarjan算法,该算法采用DFS+并查集,再看此算法之前首先你得知道并查集(尽管我相信你如果知道这个的话肯定是知道并查集的),Tarjan算法的优点在于相对稳定,时间复杂度也比较居中,也很容易理解(个人认为). 2.思想 下面详细介绍一下Tarjan算法的思想: 1.任选一个点为根节点,从根节点开始. 2.遍历该点u所有子节点v,并标记这些子节点v已被访问过. 3.若是v还有子节点,返回2,否则下一步. 4.合并v到u上. 5.寻找与当前点

洛谷P3375 [模板]KMP字符串匹配

To 洛谷.3375 KMP字符串匹配 题目描述 如题,给出两个字符串s1和s2,其中s2为s1的子串,求出s2在s1中所有出现的位置. 为了减少骗分的情况,接下来还要输出子串的前缀数组next.如果你不知道这是什么意思也不要问,去百度搜[kmp算法]学习一下就知道了. 输入输出格式 输入格式: 第一行为一个字符串,即为s1(仅包含大写字母) 第二行为一个字符串,即为s2(仅包含大写字母) 输出格式: 若干行,每行包含一个整数,表示s2在s1中出现的位置 接下来1行,包括length(s2)个整

洛谷.3803.[模板]多项式乘法(FFT)

题目链接:洛谷.LOJ. FFT相关:快速傅里叶变换(FFT)详解.FFT总结.从多项式乘法到快速傅里叶变换. #include <cmath> #include <cctype> #include <cstdio> #include <algorithm> #define gc() getchar() const int N=1e6+5; const double PI=acos(-1); int n,m; struct Complex { double

洛谷.1919.[模板]A乘B Problem升级版(FFT)

题目链接:洛谷.BZOJ2179 //将乘数拆成 a0*10^n + a1*10^(n-1) + ... + a_n-1的形式 //可以发现多项式乘法就模拟了竖式乘法 所以用FFT即可 注意处理进位 //n位*n位最多就只有2n位了 //论putchar的速度..还是快的 #include <cmath> #include <cstdio> #include <cctype> #include <algorithm> #define gc() getchar

洛谷 P3804 【模板】后缀自动机

来一份模板 1 #include<cstdio> 2 #include<algorithm> 3 #include<cstring> 4 #include<queue> 5 using namespace std; 6 typedef long long LL; 7 char s[1000100]; 8 int n; 9 LL ans; 10 namespace SAM 11 { 12 int mem,np,root; 13 int len[2000100]

[模板]后缀自动机

传送门 Description 给定一个只包含小写字母的字符串\(S\), 请你求出 \(S\) 的所有出现次数不为 \(1\) 的子串的出现次数乘上该子串长度的最大值. Solution 保持好习惯吧,模板题还是放一下 SAM的板子,想必是到处都有,反正都比我写的好看... 当初想学SAM的时候,就被某俄文翻译的\(20000\)字论文吓跑了... 核心? 作为一种可以表示所有后缀的状态的自动机,它得满足状态数尽可能的小 SAM的做法: 每个状态表示所有\(Right\)集合相同的子串,这里\

洛谷 [P2483] [模板] k短路

人生中的第一道黑题... 其实就是k短路模板 #include <iostream> #include <cstdio> #include <cstring> #include <cmath> #include <algorithm> #include <cstdlib> #include <queue> using namespace std; const int MAXN=400005; int init(){ int