[模板] 回文树/回文自动机 && BZOJ3676:[Apio2014]回文串
回文树/回文自动机
放链接:
状态数的线性证明
并没有看懂上面的证明,所以自己脑补了一个...
引理: 每一个回文串都是字符串某个前缀的最长回文后缀.
证明. 考虑一个回文串在字符串中第一次出现的位置, 记为 \(S_{p_1 ... p_2}\), 它一定是 \(S_{1 ... p_2}\)的最长回文后缀.
否则, 如果有 \(S_{p_3 ... p_2} (p_3<p_1)\) 也为回文串, 那么由于回文, \(S_{p_3 ... p_3-p_2+p_1} = S_{p_1 ... p_2}\), \(S_{p_1 ... p_2}\)并不是它第一次出现的位置.矛盾.
因而命题得证.
而每个点的最长回文后缀是唯一的, 因此\(S\)最多只有\(|S|\)个不同的回文子串.
引理的推论. 一个回文串 \(\leftrightarrow\) 某个串的最长回文子串 && 某个串的最长回文子串的回文后缀.
关于fail指针
fail指针指向的是一个节点代表的回文串的最长回文后缀.
在build时, 它也可以理解为以某个点为结尾的次长回文后缀.
Code
const int ssz=300050;
ll n;
char s[ssz];
struct te{int l,fail,cnt,ch[27];}tree[ssz]{{0,1},{-1,1}};
int pt=1,rt0=0,rt1=1;
#define ch(p,c) tree[p].ch[c]
#define fail(p) tree[p].fail
int newnd(){return ++pt;}
int getfail(int p,int i){
while(s[i-1-tree[p].l]!=s[i])p=fail(p);
return p;
}
void build(){
int p,q,last=0;
rep(i,1,n){
p=getfail(last,i);
if(ch(p,s[i])==0){
q=newnd();
tree[q].l=tree[p].l+2,fail(q)=ch(getfail(fail(p),i),s[i]);
ch(p,s[i])=q;
}
last=ch(p,s[i]);
++tree[last].cnt;
}
}
应用
枚举所有回文子串
dfs即可.
拓扑序
显然拓扑序就是 ${ 1, 2, \cdots, n } $.
求字符串出现次数
加入每个字符后, ++cnt[last];;
然后逆拓扑序dp, cnt[fa(p)] += cnt[p].
cnt[p] 即为回文串 \(p\) 出现次数.
详见下面的题.
每个节点长度 \(\le \frac {len}2\) 的回文后缀
和维护fail指针大体类似, 加上限制条件即可.
详见代码.
其中tree[p].tr表示的是 \(p\) 节点长度 \(\le \frac {len}2\) 的回文后缀
struct tnd{int l,fi,ch[csz],tr;}tree[ssz];
#define ch(p,c) tree[p].ch[c]
#define fail(p) tree[p].fi
#define trl(p) tree[p].l
#define trtr(p) tree[p].tr
int rt0=0,rt1=1,pt=1;
int getfail(int p,int i){
while(s[i-1-trl(p)]!=s[i])p=fail(p);
return p;
}
void build(){
int p,q,last=0;
rep(i,1,n){
p=getfail(last,i);
if(ch(p,s[i])==0){
q=++pt;
trl(q)=trl(p)+2,fail(q)=ch(getfail(fail(p),i),s[i]);
//get tr(p) start
if(trl(q)<=1)trtr(q)=fail(q);
else{
int z=trtr(p);
while(s[i-1-trl(z)]!=s[i]||(trl(z)+2)*2>trl(q))z=fail(z);
trtr(q)=ch(z,s[i]);
}
//end
ch(p,s[i])=q;
}
last=ch(p,s[i]);
}
}
例题
BZOJ3676:[Apio2014]回文串
求回文串长度*出现次数的最大值.
板子题.
#include<cstdio>
#include<iostream>
#include<cmath>
#include<cstring>
#include<algorithm>
#include<set>
#include<map>
using namespace std;
#define rep(i,l,r) for(register int i=(l);i<=(r);++i)
#define repdo(i,l,r) for(register int i=(l);i>=(r);--i)
#define il inline
typedef double db;
typedef long long ll;
//---------------------------------------
const int ssz=300050;
ll n,ans=0;
char s[ssz];
struct te{int l,fail,cnt,ch[27];}tree[ssz]{{0,1},{-1,1}};
int pt=1,rt0=0,rt1=1;
#define ch(p,c) tree[p].ch[c]
#define fail(p) tree[p].fail
int newnd(){return ++pt;}
int getfail(int p,int i){
while(s[i-1-tree[p].l]!=s[i])p=fail(p);
return p;
}
void build(){
int p,q,last=0;
rep(i,1,n){
p=getfail(last,i);
if(ch(p,s[i])==0){
q=newnd();
tree[q].l=tree[p].l+2,fail(q)=ch(getfail(fail(p),i),s[i]);
ch(p,s[i])=q;
}
last=ch(p,s[i]);
++tree[last].cnt;
}
}
int main(){
ios::sync_with_stdio(0),cin.tie(0);
cin>>(s+1);
n=strlen(s+1);
rep(i,1,n)s[i]-='a'-1;
build();
repdo(i,pt,2){
tree[fail(i)].cnt+=tree[i].cnt;
ans=max(ans,(ll)tree[i].cnt*tree[i].l);
}
cout<<ans<<'\n';
return 0;
}
[模板] 回文树/回文自动机 && BZOJ3676:[Apio2014]回文串的更多相关文章
- bzoj3676 [Apio2014]回文串 卡常+SAM+树上倍增
bzoj3676 [Apio2014]回文串 SAM+树上倍增 链接 bzoj luogu 思路 根据manacher可以知道,每次暴力扩展才有可能出现新的回文串. 所以推出本质不同的回文串个数是O( ...
- [Bzoj3676][Apio2014]回文串(后缀自动机)(parent树)(倍增)
3676: [Apio2014]回文串 Time Limit: 20 Sec Memory Limit: 128 MBSubmit: 3396 Solved: 1568[Submit][Statu ...
- [BZOJ3676][APIO2014]回文串(Manacher+SAM)
3676: [Apio2014]回文串 Time Limit: 20 Sec Memory Limit: 128 MBSubmit: 3097 Solved: 1408[Submit][Statu ...
- HDU - 5157 :Harry and magic string (回文树,求多少对不相交的回文串)
Sample Input aca aaaa Sample Output 3 15 题意: 多组输入,每次给定字符串S(|S|<1e5),求多少对不相交的回文串. 思路:可以用回文树求出以每个位置 ...
- HDU5658:CA Loves Palindromic (回文树,求区间本质不同的回文串数)
CA loves strings, especially loves the palindrome strings. One day he gets a string, he wants to kno ...
- bzoj千题计划304:bzoj3676: [Apio2014]回文串(回文自动机)
https://www.lydsy.com/JudgeOnline/problem.php?id=3676 回文自动机模板题 4年前的APIO如今竟沦为模板,,,╮(╯▽╰)╭,唉 #include& ...
- 【回文自动机】bzoj3676 [Apio2014]回文串
回文自动机讲解!http://blog.csdn.net/u013368721/article/details/42100363 pam上每个点代表本质不同的回文子串.len(i)代表长度,cnt(i ...
- BZOJ3676[Apio2014]回文串——回文自动机
题目描述 考虑一个只包含小写拉丁字母的字符串s.我们定义s的一个子串t的“出 现值”为t在s中的出现次数乘以t的长度.请你求出s的所有回文子串中的最 大出现值. 输入 输入只有一行,为一个只包含小写字 ...
- BZOJ3676 APIO2014回文串(manacher+后缀自动机)
由于本质不同的回文子串数量是O(n)的,考虑在对于每个回文子串在第一次找到它时对其暴力统计.可以发现manacher时若右端点移动则找到了一个新回文串.注意这样会漏掉串长为1的情况,特判一下. 现在问 ...
随机推荐
- 前端导出excel数据-jsonToExcel
咳咳,好久没有写博了... 在工作中遇到了纯前端,将数据导出为excel文件.正文开始: 第一步 安装依赖: npm i xlsx 第二步 写导出函数: import XLSX from 'xlsx' ...
- O(n*logn)级别的算法之二(快速排序)的三种实现方法详解及其与归并排序的对比
一,单路快排1.测试用例: #ifndef INC_06_QUICK_SORT_DEAL_WITH_NEARLY_ORDERED_ARRAY_SORTTESTHELPER_H #define INC_ ...
- Selenium自动化测试-进阶2-框架篇
前面的文章已经讲述了 Selenium自动化的入门知识,不知道各位看官看懂没有,有不懂的请留言. 接下来,开始讲解 Selenium自动化测试进阶知识. 首先讲解:自动化测试的核心: 利用自动化程序代 ...
- Linux系统使用
linux(操作系统的内核) 浏览器功能:(内核的解释) 各个浏览器 实现的方式不一样 呈现内容 //解析内容和样式 用—webkit— (内核)解析 实现交互逻辑 v8 引擎 (内核) 实现 =&g ...
- AI时代大点兵-国内外知名AI公司2018年最新盘点
AI时代大点兵-国内外知名AI公司2018年最新盘点 导言 据腾讯研究院统计,截至2017年6月,全球人工智能初创企业共计2617家.美国占据1078家居首,中国以592家企业排名第二,其后分别是英国 ...
- centos7 + python 2.7 + pip + openvswitch 杂项问题
问题1: virtual box 安装centos7 后,网口无ip, 解决方法是:配置网口上电后,默认状态为down, 修改“onboot=yes”, 修改后保存配置重启系统. 2. 安装pip的 ...
- 原生Js交互之DSBridge
文章链接:https://mp.weixin.qq.com/s/Iqd0dKM-ZW4UwkIgSTnvYg 在上一篇文章「android 记一次富文本加载之路」中 介绍了关于android加载富文本 ...
- Python IO编程
IO在计算机中指Input/Output,也就是输入和输出 一.文件读写 1.读文件 >>> f = open('/Users/michael/test.txt', 'r') --- ...
- Go-Ethereum 1.7.2 结合 Mist 0.9.2 实现代币智能合约的实例
目录 目录 1.什么是 Mist 2.Mist 在哪里下载? 3.Mist 有哪些依赖? 4.如何安装 Mist? 4.1.安装 Mist 依赖工具包 4.2.安装 Mist 4.3.启动 Mist, ...
- Oracle根据符合条件的数据循环批量更新
--批量对符合条件的表记录进行更新 --aa代表查询出的符合条件数据的别名 --aa后的表示需要符合的条件 --loop后开始写更新操作 begin for aa in (select a.objec ...