普通 \(\text{SAM Code}\)

#include <cstdio>
#include <iostream>
#include <cstring>
#define IN inline
using namespace std; template <typename T>
IN void read(T &x) {
x = 0; char ch = getchar(); int f = 0;
for(; !isdigit(ch); f = (ch == '-' ? 1 : f), ch = getchar());
for(; isdigit(ch); x = (x<<3)+(x<<1)+(ch^48), ch = getchar());
if (f) x = ~x + 1;
} typedef long long LL;
const int N = 2e6 + 5;
int n;
char str[N];
LL ans; struct SAM {
int fa[N], len[N], ch[N][26], lst, sz[N], size;
IN SAM(){size = lst = 1;}
IN void insert(int c) {
int p = lst, np = lst = ++size;
len[np] = len[p] + 1, sz[np] = 1;
for(; p && !ch[p][c]; p = fa[p]) ch[p][c] = np;
if (!p) {fa[np] = 1; return;}
int q = ch[p][c];
if (len[q] == len[p] + 1) {fa[np] = q; return;}
int nq = ++size;
for(int i = 0; i < 26; i++) ch[nq][i] = ch[q][i];
len[nq] = len[p] + 1, fa[nq] = fa[q], fa[np] = fa[q] = nq;
for(; p && ch[p][c] == q; p = fa[p]) ch[p][c] = nq;
}
int h[N], tot;
struct edge{int to, nxt;}e[N];
IN void add(int x, int y){e[++tot] = edge{y, h[x]}, h[x] = tot;}
void buildPT() {
for(int i = 2; i <= size; i++) add(fa[i], i);
}
void dfs(int x) {
for(int i = h[x]; i; i = e[i].nxt) dfs(e[i].to), sz[x] += sz[e[i].to];
if (sz[x] > 1) ans = max(ans, (LL)sz[x] * len[x]);
}
}T; int main() {
scanf("%s", str + 1), n = strlen(str + 1);
for(int i = 1; i <= n; i++) T.insert(str[i] - 'a');
T.buildPT(), T.dfs(1), printf("%lld\n", ans);
}

广义 \(\text{SAM Code}\)

#include <cstdio>
#include <iostream>
#include <cstring>
#include <queue>
#define IN inline
using namespace std; template <typename T>
IN void read(T &x) {
x = 0; char ch = getchar(); int f = 0;
for(; !isdigit(ch); f = (ch == '-' ? 1 : f), ch = getchar());
for(; isdigit(ch); x = (x<<3)+(x<<1)+(ch^48), ch = getchar());
if (f) x = ~x + 1;
} typedef long long LL;
const int N = 2e6 + 5;
int n, id[N];
char str[N];
LL ans; struct SAM {
int fa[N], len[N], ch[N][26], size;
IN SAM(){size = 1;}
IN void insert(int lst, int c, int now) {
int p = lst, np = ++size; id[now] = np;
len[np] = len[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 (len[q] == len[p] + 1) fa[np] = q;
else {
int nq = ++size;
for(int i = 0; i < 26; i++) ch[nq][i] = ch[q][i];
len[nq] = len[p] + 1, fa[nq] = fa[q], fa[np] = fa[q] = nq;
for(; p && ch[p][c] == q; p = fa[p]) ch[p][c] = nq;
}
}
ans += len[np] - len[fa[np]];
}
}S; struct Trie {
int size = 1, tr[N][26], fa[N];
IN void insert() {
int u = 1, m = strlen(str + 1);
for(int i = 1; i <= m; i++) {
int c = str[i] - 'a';
if (!tr[u][c]) tr[u][c] = ++size;
fa[tr[u][c]] = u, u = tr[u][c];
}
}
queue<int> Q;
IN void bfs() {
Q.push(1), id[1] = 1;
while (!Q.empty()) {
int now = Q.front(); Q.pop();
for(int i = 0; i < 26; i++)
if (tr[now][i]) S.insert(id[now], i, tr[now][i]), Q.push(tr[now][i]);
}
}
}T; int main() {
read(n);
for(int i = 1; i <= n; i++) scanf("%s", str + 1), T.insert();
T.bfs(), printf("%lld\n", ans);
}

P3804 【模板】后缀自动机 (SAM) && P6139 【模板】广义后缀自动机(广义 SAM)的更多相关文章

  1. 后缀自动机(SAM)+广义后缀自动机(GSA)

    经过一顿操作之后竟然疑似没退役0 0 你是XCPC选手吗?我觉得我是! 稍微补一点之前丢给队友的知识吧,除了数论以外都可以看看,为Dhaka和新队伍做点准备... 不错的零基础教程见 IO WIKI ...

  2. BZOJ 3473: 字符串 [广义后缀自动机]

    3473: 字符串 Time Limit: 20 Sec  Memory Limit: 256 MBSubmit: 354  Solved: 160[Submit][Status][Discuss] ...

  3. POJ3294Life Forms(广义后缀自动机)(后缀数组+二分+数状数组)

    You may have wondered why most extraterrestrial life forms resemble humans, differing by superficial ...

  4. CodeForces-204E:Little Elephant and Strings (广义后缀自动机求出现次数)

    The Little Elephant loves strings very much. He has an array a from n strings, consisting of lowerca ...

  5. BZOJ4566 Haoi2016 找相同字符【广义后缀自动机】

    Description 给定两个字符串,求出在两个字符串中各取出一个子串使得这两个子串相同的方案数.两个方案不同当且仅当这两 个子串中有一个位置不同. Input 两行,两个字符串s1,s2,长度分别 ...

  6. 【BZOJ3227】串【广义后缀自动机】

    题意 给出n个字符串,问每个字符串中有多少子串是这所有的n个字符串中至少k个的子串. 分析 广义后缀自动机模板题.对这n个串建广义后缀自动机,对于每个状态维护两个值cou[u]和lcu[u]分别代表拥 ...

  7. SPOJ8093Sevenk Love Oimaster(广义后缀自动机)

    Oimaster and sevenk love each other.     But recently,sevenk heard that a girl named ChuYuXun was da ...

  8. bzoj3926: [Zjoi2015]诸神眷顾的幻想乡 对[广义后缀自动机]的一些理解

    先说一下对后缀自动机的理解,主要是对构造过程的理解. 构造中,我们已经得到了前L个字符的后缀自动机,现在我们要得到L+1个字符的后缀自动机,什么需要改变呢? 首先,子串$[0,L+1)$对应的状态不存 ...

  9. BZOJ 3926 && ZJOI 2015 诸神眷顾的幻想乡 (广义后缀自动机)

    3926: [Zjoi2015]诸神眷顾的幻想乡 Time Limit: 10 Sec Memory Limit: 512 MB Description 幽香是全幻想乡里最受人欢迎的萌妹子,这天,是幽 ...

  10. BZOJ 3277 串 (广义后缀自动机)

    3277: 串 Time Limit: 10 Sec Memory Limit: 128 MB Submit: 309 Solved: 118 [Submit][Status][Discuss] De ...

随机推荐

  1. JavaEE Day06 JDBC连接池&JDBCTemplate

    今日内容: 数据库连接池 简化操作--Spring JDBC提供的 JDBC Template(JDBC的封装) 一.数据库连接池 1.引入 之前:每一次都要获取连接.释放连接-- 现在:连接重复使用 ...

  2. MyBatis03:连接池及事务控制、xml动态SQL语句、多表操作

    今日内容: mybatis中的连接池.事务控制[原理了解,应用会用] mybatis中连接池的使用及分析 mybatis中事务控制的分析 mybatis中基于xml配置的动态SQL语句使用[会用即可] ...

  3. 【每日一题】【BFS&Lambda&重建二叉树】2022年2月15日-根据先序中序重建并输出二叉树的右视图

    描述 请根据二叉树的前序遍历,中序遍历恢复二叉树,并打印出二叉树的右视图 思路:重建&层次遍历记录最后一个&Lambda表达式 答案: import java.util.*; publ ...

  4. Day38:Lambda表达式

    Lambda表达式 1.1 概述 Lambda是JDK8开始后的一种新语法形式. 作用:简化函数式匿名内部类的代码写法. 简化格式: /*部类被重写方法的参数)->{ 被重写方法的方法体代码 } ...

  5. python 集合常用操作

    集合的特性 无序.不重复.可迭代 常用api 创建一个集合 需要显式地使用set()方法来声明,如果使用字面量{}来声明解析器会认为这是一个字典. add() 往集合中添加一个元素 demo = se ...

  6. go语言行为(方法)的两种定义差别

    概述: go在定义方法时,有如下两种表示形式: 第一种,在实例方法被调用时,会产生值复制 func (e Employee) String() string {} 第二种,不会进行内存拷贝,所以通常情 ...

  7. java注解基础知识整理

    目录 1.注解的定义 1.1.定义一个注解 1.2.注解的使用 2.JDK内置注解 2.1.java.lang包下的注释类型 2.2.元注解 2.3.Deprecated注解 3.在注解中定义属性 3 ...

  8. MSIC总结取证分析——日志分析

    MSIC总结取证分析 一.日志分析: 1.常见日志分析类型: 2.常见一些考点: (1)还原特定IP攻击手段(SQL注入.暴力破解.命令执行等),或还原最初攻击时间: (2)寻找flag或者特定文件解 ...

  9. 【FAQ】推送服务常见问题及解决方案

    一.推送成功收不到消息,推送返回:{"message":"success","requestID":"1523868*****28 ...

  10. P5687 [CSP-S2019 江西] 网格图

    题面 给定一个 \(n\times m\) 的网格图,行从 \(1\sim n\) 编号,列从 \(1\sim m\) 编号,每个点可用它所在的行编号 \(r\) 与所在的列编号 \(c\) 表示为 ...