比KMP简单的Manacher

P3805 【模板】manacher - 洛谷 | 计算机科学教育新生态 (luogu.com.cn)

“没时间悼念KMP了,接下来上场的是Manacher!”


什么是Manacher?

历史背景:

1975 年,一个叫 Manacher 的人发明了这个算法,所以叫Manacher 算法(中文名:马拉车算法)

应用背景:

现在有个字符串A,你需要找到该字符串中最长的回文字串,输出其长度。

回文串是正着读反着读都是一样的,例如:101,aba,tnt。

Manacher算法思路:

Manacher第一步修改原字符串。

我们认定一个字符串是否为回文串,都会先找这个字符串的中心,然后向两边扩散进行字符对比。比如aba这个字符串,我们要判断它是否为回文串,就可以先找到中心位置b,然后再向两边扩散比较,b左边的字符是不是等于b右边的字符,如果都一致,那么该字符串就是回文串。

当然,聪明的你一定发现了一个问题,如果该字符串是abba这种偶数长度的字符串呢?这个时候我们就可以把偶数长度的字符串变为奇数长度的字符串,向原字符串插入一个原字符串不会出现的字符,比如,"$"。那么对于abba而言插入后的字符串就变成了$a$b$b$a$。然后我们就可以套用判断奇字符串是否是回文串的思想,来对这个进行判断了。

插入相同字符后的字符串如果是回文串,那么没有插入相同字符的字符串也是回文串,可自行验证。

所以Mannacher的第一步就是在原来的字符串的基础上,插入一个原字符串不会出现的字符。

Manacher第二步回文范围[L,R],回文中心W。

约定p数组放的是回文半径,则p[i]存放的是以i为回文中心的回文半径。

如图,在字符串(黑色框)中有一回文子串(蓝色框),范围为[L, R],其中M为该回文串的中心,现在我们欲求以i为回文中心的回文半径r。就要先分两种大情况。

第一种是i在[L, R]区间里,我们就可以先找i相当于M的对称点K,因为i是在K之后的所以K的回文半径是已知的,那么当K的回文区域不包含L,即K-p[K]+1>L时,p[i] = p[k] =[2*M-i]。当K的回文区域包含L时,即K-p[K]+1<=L时,p[i] = R-i+1。

第二种是i在[L, R]区间外面的,我们就只能暴力向两边扩散比较,求得它的回文半径。

经过如图的数学推导,我们就可以给第一种情况化简为p[i]=min(p[2*M-i], R-i+1),然后再向两边扩散比较字符。

通过以上这种方法,写一个for循环加一层while就能够快捷地得到以每个字符作为回文中心的回文半径了。

得到回文半径有什么用呢?题目不是让我们求字符串里面最长的回文子串有多长嘛?那么最大的回文半径-1,不就是最长的回文子串的长度嘛?也就解开了。

因为原字符串在插入了原字符串没有的字符后,原字符串的长度变成了原来的2倍,所以回文半径-1就是最长的回文子串。可自己写几个试试。

【算法详解】Manacher算法_哔哩哔哩_bilibili 


如何实现Manacher代码?

插入原字符串没有的字符

for (ll i = 0; i < 2 * a.length(); i++){if (i & 1)
//如果是奇数b[i] = a[i / 2];elseb[i] = '$';}b[2 * a.length()] = '$';
//a为原字符串
//b为插入后的字符串

求以每个字符作为回文中心的回文半径

ll M = 0, R = 0;
//初始化M和Rfor (ll i = 0; i <= 2 * a.length(); i++){
// for循环遍历字符串里面每个字符if (i > R)p[i] = 1;
//如果i是在[L, R]区间外面的 
//那么它的回文半径默认从1开始elsep[i] = min(p[2 * M - i], R - i + 1);
//否则回文半径就是min(p[2 * M - i], R - i + 1)while (i + p[i]<= 2 * a.length() && i - p[i] >= 0 && b[i - p[i]] == b[i + p[i]])
//while循环就是向两边扩散比较的过程p[i]++;
//只有当左右两边的字符一致
//回文半径才+1if (i + p[i] - 1 > R){M = i;R = i + p[i] - 1;}
//如果以新的字符为回文中心
//回文半径更大的话
//那么就以新的字符为回文中心
//更新R}

全部代码

#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;
typedef long long ll;
const ll N = 2e7;
string a;
vector<ll> p(2 * N);
char b[2 * N];int main()
{cin >> a;for (ll i = 0; i < 2 * a.length(); i++){if (i & 1)b[i] = a[i / 2];elseb[i] = '$';}b[2 * a.length()] = '$';// cout << "变形后的:" << b << endl;ll M = 0, R = 0;for (ll i = 0; i <= 2 * a.length(); i++){if (i > R)p[i] = 1;elsep[i] = min(p[2 * M - i], R - i + 1);while (i + p[i]<= 2 * a.length() && i - p[i] >= 0 && b[i - p[i]] == b[i + p[i]])p[i]++;if (i + p[i] - 1 > R){M = i;R = i + p[i] - 1;}}// cout << M << " " << R << endl;cout << *max_element(p.begin(), p.end()) - 1;return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/784996.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

财务管理系统的设计与实现|Springboot+ Mysql+Java+ B/S结构(可运行源码+数据库+设计文档)

本项目包含可运行源码数据库LW&#xff0c;文末可获取本项目的所有资料。 推荐阅读100套最新项目持续更新中..... 2024年计算机毕业论文&#xff08;设计&#xff09;学生选题参考合集推荐收藏&#xff08;包含Springboot、jsp、ssmvue等技术项目合集&#xff09; 目录 1. …

02-JDK新特性-接口新特性

接口新特性 接口组成更新概述 接口的组成 常量 public static final String ZERO "0";抽象方法 public abstract void dance();默认方法&#xff08;JAVA8新增&#xff09; public default void dance(){}静态方法&#xff08;JAVA8新增&#xff09; public stat…

leecode 331 |验证二叉树的前序序列化 | gdb 调试找bug

计算的本质是数据的计算 数据的计算需要采用格式化的存储&#xff0c; 规则的数据结果&#xff0c;可以快速的按照指定要求存储数据 这里就不得不说二叉树了&#xff0c;二叉树应用场景真的很多 本题讲的是&#xff0c;验证二叉树的前序序列化 换言之&#xff0c;不采用建立树的…

Logback日志框架常见配置

版权声明 本文原创作者&#xff1a;谷哥的小弟作者博客地址&#xff1a;http://blog.csdn.net/lfdfhl logback简介 Logback是一个高性能、功能强大的日志框架&#xff0c;专为Java应用程序设计。它由Log4j的创始人Ceki Glc创建&#xff0c;并被视为Log4j的继承者和改进版。Lo…

3D Web轻量化平台HOOPS Web Platform在大型水利工程中的应用和价值

随着科技的不断进步和互联网的普及&#xff0c;数字化技术在各个领域的应用日益广泛&#xff0c;大型水利工程也不例外。作为一种先进的3D Web轻量化引擎&#xff0c;HOOPS Web Platform在大型水利工程中扮演着重要的角色&#xff0c;为工程设计、施工管理、运行维护等方面提供…

Ubuntu部署BOA服务器

BOA服务器概述 BOA是一款非常小巧的Web服务器&#xff0c;源代码开放、性能优秀、支持CGI通用网关接口技术&#xff0c;特别适合用在嵌入式系统中。 BOA服务器主要功能是在互联嵌入式设备之间进行信息交互&#xff0c;达到通用网络对嵌入式设备进行监控&#xff0c;并将反馈信…

Go的数据结构与实现【Binary Search Tree】

介绍 本文用Go将实现二叉搜索树数据结构&#xff0c;以及常见的一些方法 二叉树 二叉树是一种递归数据结构&#xff0c;其中每个节点最多可以有两个子节点。 二叉树的一种常见类型是二叉搜索树&#xff0c;其中每个节点的值都大于或等于左子树中的节点值&#xff0c;并且小…

Unbtun-arach64架构安装PySide2(python3.6)

aarch平台是无法通过pip安装PySide2的&#xff0c;同时利用源码下载一直报错 1. 我是python3.6.9&#xff0c;在官网上找到对应的PySide2版本 5.15.2.所以首先在官网下载Qt5.15.2的源码&#xff1a;https://download.qt.io/archive/qt/5.15/5.15.2/single/ 2. 编译qt环境 aar…

Seata(分布式事务实例环境搭建)

文章目录 1.基本介绍1.引出Seata2.问题分析 2.Seata的安装和配置1.解压到d盘2.修改D:\seata\conf\file.conf文件1.修改事务组2.修改日志存储模式为db3.修改数据库&#xff08;MySQL5.7&#xff09;连接信息4.创建seata数据库5.复制db_store.sql的内容&#xff0c;创建需要的表6…

Web实例_报表开发01-基于HTML进行报表呈现

Web实例_报表开发01-基于HTML进行报表呈现 报表开发是一种在利用了软件的基础上, 针对不同类型的报表, 进行开放的工作。 而以报表的方式, 将相关的内容、数值呈现出来的话, 则会起到更好的概况作用。 再加上, 报表开发工作是依托于计算机来完成的, 因此在效率、完整性等方面…

红酒:分类视角下的红酒品质评估与标准制定

在红酒的世界中&#xff0c;品质的评估与标准的制定对于维护消费者权益、促进行业健康发展具有重要意义。云仓酒庄雷盛红酒作为业界持续发展品牌&#xff0c;从分类的视角出发&#xff0c;对红酒品质进行了多方的评估&#xff0c;并积极参与制定相关的标准。 首先&#xff0c;从…

优思学院|工程经理应如何利用PDCA循环?

作为工程经理&#xff0c;理解PDCA&#xff08;计划-执行-检查-行动&#xff09;质量管理循环程序对于确保项目质量和持续改进是十分重要。 PDCA 最早由美国质量管理专家 Walter A. Shewhart (1939)提出&#xff0c;其后被戴明博士&#xff08;Dr. Deming&#xff09;所采用、…

【Linux】权限理解

权限理解 1. shell命令以及运行原理2. Linux权限的概念3. Linux权限管理3.1 文件访问者的分类&#xff08;人&#xff09;3.2 文件类型和访问权限&#xff08;事物属性&#xff09;3.2.1 文件类型3.2.2 基本权限 3.3 文件权限值的表示方法3.4 文件访问权限的相关设置方法3.4.1 …

基于energy score的out-of-distribution数据检测,LeCun都说好 | NerulPS 2020

论文提出用于out-of-distributions输入检测的energy-based方案&#xff0c;通过非概率的energy score区分in-distribution数据和out-of-distribution数据。不同于softmax置信度&#xff0c;energy score能够对齐输入数据的密度&#xff0c;提升OOD检测的准确率&#xff0c;对算…

【JAVA】java基础(分支结构)洛谷刷题(含图解:吃苹果)

题目&#xff1a; 解析 代码 import java.util.Scanner;public class Main {public static void main(String[] args) {// TODO Auto-generated method stubScanner input new Scanner(System.in);int apple,time,rate; appleinput.nextInt(); rateinput.nextInt(); timeinpu…

专题【链表】刷题日记

题目列表 学习题(22题) 2024.03.31 两数相加 19. 删除链表的倒数第 N 个结点 合并K个升序链表 2024.04.01 24. 两两交换链表中的节点 25. K 个一组翻转链表 61. 旋转链表 83. 删除排序链表中的重复元素 82. 删除排序链表中的重复元素 II 86. 分隔链表 92. 反转链表 II 1…

【STM32嵌入式系统设计与开发】——14PWM(pwm脉宽输入应用)

这里写目录标题 一、任务描述二、任务实施1、WWDG工程文件夹创建2、函数编辑&#xff08;1&#xff09;主函数编辑&#xff08;2&#xff09;USART1初始化函数(usart1_init())&#xff08;3&#xff09;USART数据发送函数&#xff08; USART1_Send_Data&#xff08;&#xff09…

服务器安全事件应急响应排查方法

针对服务器操作系统的安全事件也非常多的。攻击方式主要是弱口令攻击、远程溢出攻击及其他应用漏洞攻击等。分析安全事件&#xff0c;找到入侵源&#xff0c;修复漏洞&#xff0c;总结经验&#xff0c;避免再次出现安全事件&#xff0c;以下是参考网络上文章&#xff0c;总结的…

初识编译和链接(C语言)

文章目录 编译和链接翻译环境预处理编译汇编链接 运行环境 编译和链接 编译和链接这两个大的过程构成了翻译环境。 其实&#xff0c;在ANSI C的任何一种实现中&#xff0c;存在两个不同的环境。 一个环境是翻译环境&#xff0c;另一个是执行环境。 翻译环境中&#xff0c;源…

【软件工程】详细设计(一)

1. 引言 1.1 编写目的 该文档的目的是描述《学生成绩管理系统》项目的详细设计&#xff0c;其主要内容包括&#xff1a; 系统功能简介 系统详细设计简述 各个模块的实现逻辑 最小模块组件的伪代码 本文档的预期的读者是&#xff1a; 开发人员 项目管理人员 测试人员 …