马拉车算法
2021/7/27 22:05:50
本文主要是介绍马拉车算法,对大家解决编程问题具有一定的参考价值,需要的程序猿们随着小编来一起学习吧!
含义
就是一个\(O(n)\)的复杂度求解最长回文子串的算法
思路
思路的话我随便说下
首先回文串可能是奇数也可能是偶数,那么对称中心就有可能是两个字符的空隙,所以先给每个字符插如一个隔板符号 '|' 第0个字符插入'~' 防止出现超出边界的问题
如abcbs -> ~|a|b|c|b|s|
设\(p[i]\)以\(i\)为中点的回文半径包括自己本身
例如|a|b|a|
那么\(p[4]=4\)
我们也维护一个$ mx$ 和 \(id\),表示对于当前计算的\([1,i-1]\)中,\(i + p[i]\) 的最大值是$ mx\(,\)mx \(对应的\) i \(记为\)id$
当你现在开始计算 p[i] 时,默认$ p[1..i-1] $都已经算出。如果 \(mx > i\),那么根据对称得出
\(p[i] >= min(p[2 \times id - i], mx - i+1)\)
由下图可以看出
而最后的答案即为\(max(p[i])-1\)
很多人都没有解释为什么,其实很简单
根据\(p[i]\)的定义,那么扩展的回文串的长度即为\(2p[i]-1\)
而其中#
占\(p[i]\)个,所以原回文串的长度即为\(p[i]-1\)
这个算法挺简单的,感觉比kmp都要简单很多
模板题
代码
#include<bits/stdc++.h> using namespace std; typedef long long ll; typedef unsigned long long ull; //typedef pair<int,int> pii; #define fi first #define se second #define debug printf("aaaaaaaaaaa\n"); const int maxn=2e7+5,inf=0x3f3f3f3f,mod=998244353; const ll INF=0x3f3f3f3f3f3f3f3f; const double eps=1e-7; char temp[maxn]; char s[maxn]; int p[maxn]; int main(){ scanf("%s",temp+1); int n=strlen(temp+1); n=2*n+1; s[0]='~'; for(int i=1;i<=n;i++){ if(i%2){ s[i]='|'; }else{ s[i]=temp[i/2]; } } int mx=0,id=1,ans=0; for(int i=1;i<=n;i++){ p[i]=min(p[2*id-i],mx-i+1); while(s[i+p[i]]==s[i-p[i]]) p[i]++; if(p[i]+i-1>=mx){ mx=p[i]+i-1; id=i; ans=max(ans,p[i]-1); } } printf("%d\n",ans); return 0; }
这篇关于马拉车算法的文章就介绍到这儿,希望我们推荐的文章对大家有所帮助,也希望大家多多支持为之网!
- 2025-01-102025 蛇年,J 人直播带货内容审核团队必备的办公软件有哪 6 款?
- 2025-01-10高效运营背后的支柱:文档管理优化指南
- 2025-01-10年末压力山大?试试优化你的文档管理
- 2025-01-10跨部门协作中的进度追踪重要性解析
- 2025-01-10总结 JavaScript 中的变体函数调用方式
- 2025-01-10HR团队如何通过数据驱动提升管理效率?6个策略
- 2025-01-10WBS实战指南:如何一步步构建高效项目管理框架?
- 2025-01-10实现精准执行:团队协作新方法
- 2025-01-10如何使用工具提升活动策划团队的工作效率?几个必备工具推荐
- 2025-01-10WiX 标签使用介绍:打造专业安装程序的利器