[OI] 欢夏!邪龙?马拉车!
标题来自原神
算法概述
Manacher 算法
用途:寻找回文串,最板子的情况下用于字符串的回文子串计数
给定一个字符串 \(S\),求出它全部的回文子串
容易想到一种暴力的 \(n^{2}\) 做法,即枚举全部中心点,开双指针向两边扩展,每扩展一次就提供 \(1\) 的贡献.
事实上,对于这样的算法来说,判断偶回文串会不太方便:因为它没有中心点.
因此,我们考虑在原字符串里加入一些莫名其妙的字符,如下:
cthoiissb
c#t#h#o#i#i#s#s#b
这样做,我们发现,当枚举原字符串内的点作为中心点时,就相当于在枚举奇回文子串,当枚举特殊字符作为中心点时,就相当于在枚举偶回文子串. 我们通过上述方法完成了对奇偶子串的统一.
下面我们来考虑对进行的转移进行优化
首先我们引入一些概念:
定义回文半径表示回文串的长度除以二的值(显然,在处理之后,回文串全都变成了奇回文串,因此直接向下取整即可),我们这样定义是因为处理后的回文半径长等于处理前的回文串长度减一,如下:
cthtc length=5
c#t#h#t#c length=4
类似地,我们还可以求出该回文串在原串中的位置:
oicthtc
o#i#c#t#h#t#c length=4
[8]
可以发现,cthtc 的对称中心 h 在原字符串中的位置为 \(5\)(以下默认下标从 \(0\) 开始),在处理后的字符串中的位置为 \(8\),而回文半径长为 \(4\),考虑到 \(8+1-4=5\),发现再多举几组也是如此的性质,即 “原字符串位置等于处理后字符串位置加一减去回文半径”,考虑到这样有点麻烦,因此在处理后的字符串前插入一个不同的字符(也是为了防止在 \(p=0\) 时访问负数下标),实际上如果保守一点的话,末尾也是需要插入特殊字符,只不过因为末尾有一个 \0,因此不需要插入. 需要注意的是,如果真的要插入的话,首位字符不能相同,否则直接就把他们两个匹配上了,会影响答案.
到现在我们的算法还是 \(n^{2}\) 的,下面我们来考虑优化这种转移.

图例中的 \(T\) 表示了一种可能的字符串,下方的每一位的 \(P\) 表示了以当前位为中心的最长回文半径. 一般来说,这个 \(P\) 数组需要我们从头到尾扫一遍来求,这一点我们无从优化,我们来考虑如何才能从之前的状态跳到当前的状态.
如图,我们已经求出了一部分 \(P\) 的值,注意到图中有一个很大的 \(P_{11}=9\),我们考虑利用一下它,因此用虚线标出它的中心与左右边界. 现在我们的目标是求出 \(P_{13}\)
假如 \(P_{11}\) 左右两边对称的话,可以想到我们只需要找到需要求的点在另一边的对称点,那么对称点的 \(P\) 值一定就也是当前点的 \(P\) 值:因为回文的性质,既然对称点是回文,对称过来也一定是回文,并且因为之前求的是最大值,因此也不存在一个更大的值了,所以直接转移即可.

下面我们再来考虑另一种情况:

现在的目标是求 \(P_{15}\),按照刚才的思路,现在我们应该去找对称点 \(P_{7}\),但是我们发现此时无法直接进行转移.
刚才我们转移是建立在一个很大的回文区间 \(P_{11}=9\) 上的,因为两边的回文区间全部都在这个大的回文区间内,因此我们才能保证两边的字符是相等的. 现在区间超出了大的对称区间的范围,因此不能保证超出范围的部分是相等的,也就不能直接转移了.
不过这样的情况还是有利用价值的,考虑可以先把能保证相等的转移了,即转移到大区间的边界,剩下的没有办法,可以直接进行暴力转移,为下一次转移助力. 可以证明这样做的复杂度是 \(O(n)\) 的.

习题
P3805 模板
在 Manacher 的具体实现中,还有几点需要注意的:
- 因为我们的算法需要找到一个最优的区间,使其能尽可能覆盖我们当前要求的 \(P\),因此我们引入一个变量 \(id\) 来表示最优区间的中心. 考虑到,最优状态发生转移时,当且仅当新状态右边界比当前的最优右边界还要靠右,因为此时能更新到的边界也更加靠右.
- 引入的 \(mx\) 变量用于表示当前最优区间的右边界,其实是一个非必须变量,你完全可以用 \(id+p_{id}\) 来表示它
- 模板题要求的是回文子串的长度最大值,因此函数内最下方 if() 统计答案方式有所改变. 假如要统计子串数量,仅需改成
ans+=p[i]即可 - 请注意上述推导中,最大长度为回文半径减一
namespace manacher{
vector<int>p;
string insert(string x){
string res="/#";
for(int i=0;i<=x.length()-1;++i){
res.push_back(x[i]);
res.push_back('#');
}
res.push_back('?');
return res;
}
int act(string x){
x=insert(x);
p.resize(x.size());
int mx=0,id=0,ans=0;
for(int i=1;i<=x.length()-1;++i){
p[i]=(mx>i)?min(p[2*id-i],mx-i):1;
while(x[i+p[i]]==x[i-p[i]]){
p[i]++;
}
if(mx<i+p[i]){
mx=i+p[i];
id=i;
}
if(ans<p[i]){
ans=p[i]-1;
}
}
return ans;
}
}
串串
眼睛看到的:字符串
脑子里想到的

一个字符串 \(S\) 由 \(s\) 拼接而成,即 \(s\) 正序和倒序首尾相接(首位共用一个字符)
如abc可以拼成abcbabcba
现在给出 \(S\) 的前面若干部分,求长度小于给定部分的全部合法的 \(s\)
样例:abcdcb可以是abcd拼成的abcdcba的一部分,或者abcdcb拼成的一部分,因此共两种
考虑分三种情况讨论:
- 由长度超过给定字符串长度一半的字符串拼接:此种情况应至多拼接两次,因此只需要判断最长回文子串的末端是否到达右边界即可
- 由长度不足给定字符串长度一半的字符串拼接:此种情况应拼接了多次,只需要判断当前位为回文,并且翻转后仍能翻转即可. 这一点我们可以通过改变遍历顺序来通过标记处理.
- 可以发现其本身一定能构成一个 \(s\)
形象一点?比如给出字符串 qwqwq,它可以是由子串 qwqw 翻转而得的,而 qwqw 是由子串 qwq 翻转而得的,而 qwq 是由子串 qw 翻转而得的,然而,qw 无法由它的子串翻转而得,此时算法结束
#include<bits/stdc++.h>
using namespace std;
string x;
namespace manacher{
vector<int>p;
string insert(string x){
string res="/#";
for(int i=0;i<=x.length()-1;++i){
res.push_back(x[i]);
res.push_back('#');
}
res.push_back('?');
return res;
}
void act(string x){
p.clear();
x=insert(x);
p.resize(x.size());
int mx=0,id=0;
for(int i=1;i<=x.length()-1;++i){
p[i]=(mx>i)?min(p[2*id-i],mx-i):1;
while(x[i+p[i]]==x[i-p[i]]){
p[i]++;
}
if(mx<i+p[i]){
mx=i+p[i];
id=i;
}
}
}
}
bool vis[1000001];
int main(){
int cases;cin>>cases;while(cases--){
cin>>x;
manacher::act(x);
for(int i=x.length()*2-2;i>=2;i-=2){
vis[i/2]=false;
if((manacher::p[i]-1+i)/2==x.length()) vis[i/2]=true;
if(vis[(manacher::p[i]-1+i)/2] and manacher::p[i]==i) vis[i/2]=true;
}
for(int i=1;i<=x.length()-1;++i){
if(vis[i]){
cout<<i<<" ";
vis[i]=false;
}
}
cout<<x.length()<<endl;
}
}
[OI] 欢夏!邪龙?马拉车!的更多相关文章
- 天气预报API(二):全球城市、景点代码列表(“旧编码”)
说明 2016-12-10 补充 (后来)偶然发现中国天气网已经有城市ID列表的网页...还发现城市编码有两种,暂且称中国天气网这些编码为旧标准"旧编码"的特征是 9个字符长度; ...
- 浅析软件工程中的UML建模技术
一.基本信息 标题:浅析软件工程中的UML建模技术 时间:2018 出版源:电子世界 领域分类:软件工程:UML建模技术:需求分析 二.研究背景 问题定义:软件工程中UML建模技术的研究 难点:明确软 ...
- Ajax的初体验
一.AJAX的介绍 Ajax 即“Asynchronous Javascript And XML”(异步 JavaScript 和 XML),是指一种创建交互式网页应用的网页开发技术. Ajax = ...
- 搞笑OI
OI难 噫吁嚱,维护难哉!OI之难,难于上青天!哈希及DP,代码何茫然!尔来一千两百A,不见金牌背后难.西当华师有考场,可以横绝CN巅.编译不过壮士死,然后超时爆内存相钩连.上有自主招生之高标,下有由 ...
- 再见,OI
你好,NOIP 2015年9月1日 正式成为了福建省莆田一中的一名高一成员 后来学校搞了选修 大家都很激动 因为自己的兴趣和特长能够得到发挥了(或者说能逃课或者看好多电影) 发现选修提供的选项中有好几 ...
- 08重编终极版《东邪西毒:终极版》DVD粤语中字
1.东邪西毒].Ashes.of.Time.1994.384p.DVDRip.x264.ac3-DTMM.mkv 这个版本最清晰 ,可惜删减了,只有87分钟,粤语,1.4G. 2.东邪西毒(初始版). ...
- OI暑假集训游记
莞中OI集训游记 Written BY Jum Leon. I 又是一载夏,本蒟蒻以特长生考入莞中,怀着忐忑的心情到了8月,是集训之际.怀着对算法学习的向往心情被大佬暴虐的一丝恐惧来到了 ...
- OI生涯回忆录 2017.9.10~2018.11.11
然而并没有退役 为了这两天,也准备了一年. 高一零基础的蒟蒻,NOIP2017仅有80pts 之后看着luogu的倒计时, 300天,200天,100天,30天, 直到10天.1天. 10号,11号的 ...
- 写在OI退役后和高中毕业前的一些话
更新日志: 2017.02.13 开坑 2017.02.13 更新[零][壹] 2017.02.14 更新[贰] 2017.02.26 更新[叁][肆] 2017.03.04 锅多如狗,停更一周 20 ...
- 我的OI生涯 第六章
开学了,但是我们并没有像一个正常的高二学生一样坐在教室里接受调研考试的洗礼. 暑假作业这种东西早已被甩在一旁,可以想象回去补文化课时该有多么狼狈. 大王给我们制定了周密的计划,每周两次测试,加上蔡老师 ...
随机推荐
- [rCore学习笔记 00]总览
写在前面 本随笔是非常菜的菜鸡写的.如有问题请及时提出. 可以联系:1160712160@qq.com GitHhub:https://github.com/WindDevil (目前啥也没有 rCo ...
- 对于同一个项目,同时将其git到GitHub和Gitee
对于同一个项目,你可以同时将其git到GitHub和Gitee.这通常通过配置多个远程仓库地址来实现.以下是一步步的操作指南: 一.在GitHub和Gitee上创建仓库 GitHub: 登录GitHu ...
- CF1929B Sasha and the Drawing 题解
CF1929B 题意 给定一个 \(n\times n\) 的正方形,已知正方形最多有 \(4\times n-2\) 条对角线,要求要有至少 \(k\) 条对角线经过至少一块黑色方格,求至少要将几条 ...
- AngleScript语法
Class的使用要继承于Interface或者Mixin class.Mixinclass实际上就是类似于抽象类 ,它已经实现的,在子类里面不能实现,类似如下代码: interface AInterf ...
- holiday week3
本周开始进行小学期实验报告B 上周LOL打的rank分又掉了回去 星际争霸打到了铂金段位 JAVA预备在小学期完成之后开始正式学习 现已了解雏形 本周发布了一个视频https://www.bilibi ...
- 【Vue】MineData 地图接入
一.文档资料: MineData开放平台: https://minedata.cn/md-platform/login/login MineData V2.1.0 接口文档: http://113.1 ...
- 【FastDFS】06 SpringBoot实现上传
创建SpringBoot工程: 再导入所需要的依赖: <dependency> <groupId>net.oschina.zcx7878</groupId> < ...
- 【SqlServer】Windows-2019 安装
安装教程参考: https://blog.csdn.net/weixin_43790591/article/details/104149800 数据库SQL Server 2019 + 管理工具SQL ...
- 如何计算两个正太分布的KL散度 —— 正太分布的KL散度 (Kullback-Leibler divergence) 计算
参考: https://blog.csdn.net/int_main_Roland/article/details/124650909 给出实现代码: def get_kl(): mean0, log ...
- Apache DolphinScheduler 3.1.8 保姆级教程【安装、介绍、项目运用、邮箱预警设置】轻松拿捏!
概述 Apache DolphinScheduler 是一个分布式易扩展的可视化 DAG 工作流任务调度开源系统.适用于企业级场景,提供了一个可视化操作任务.工作流和全生命周期数据处理过程的解决方案. ...