Manacher算法

算法描述与证明

cpp 复制代码
for (int i = 1; i <= m; i++) { // 遍历预处理后的字符串t
    p[i] = 1;                 // 初始化最小回文半径为1(单个字符)
    if (maxid > i)            // 如果i在当前最右回文边界内
        p[i] = min(p[2 * id - i], maxid - i); // 利用对称性快速确定下限
    
    // 暴力扩展回文半径
    while (t[i - p[i]] == t[i + p[i]]) 
        p[i]++;               // 若两侧字符相等,半径+1
    
    // 更新最右回文边界及中心
    if (i + p[i] > maxid) {
        id = i;
        maxid = i + p[i];
    }
}

部分代码分析

cpp 复制代码
while(t[i-p[i]] == t[i+p[i]])
    p[i]++;

这段代码是看在当前以 i 为中心(如果超出了)最大范围外的左边和右边是否相等,如果相等说明可以进一步扩展,将当前回文半径加一。

cpp 复制代码
if(maxid > i)
    p[i] = min(p[2 * id - i], maxid - i);

这段代码是,如果当前点在当前最大回文子串的范围内,找当前点的以当前最大回文子串中心点的对称点(以当前最大回文子串中心点折过去所对应的点)和当前点到当前最大回文子串右边界的距离的最小值。因为串对称过来以后超过当前最大回文子串右边界的话,超出的部分就不能对称过来了(不能保证左侧超出的部分和右侧超出的部分相同)。所以要取两者最小值。

完整流程

步骤 操作 目的
1 预处理字符串(插入#和哨兵) 统一奇偶回文为奇数长度
2 初始化p[i]=1 每个字符至少自身是回文
3 利用对称性快速确定下限p[i] = min(p[j], maxid-i) 跳过重复比较,优化效率
4 暴力扩展while(...) p[i]++ 验证并扩大回文半径
5 更新maxidid 维护当前最右回文边界信息
相关推荐
闻缺陷则喜何志丹1 小时前
【树的直径 离散化】 P7807 魔力滋生|普及+
c++·算法·洛谷·离散化·树的直径
AI_Ming1 小时前
Seq2Seq-大模型知识点(程序员转行AI大模型学习)
算法·ai编程
若水不如远方1 小时前
分布式一致性(六):拥抱可用性 —— 最终一致性与 Gossip 协议
分布式·后端·算法
计算机安禾1 小时前
【C语言程序设计】第35篇:文件的打开、关闭与读写操作
c语言·开发语言·c++·vscode·算法·visual studio code·visual studio
Wect2 小时前
React Hooks 核心原理
前端·算法·typescript
美式请加冰2 小时前
字符串的介绍和使用
算法
不想写代码的星星2 小时前
告别 C 风格枚举:为什么你应该使用 enum class
c++
m0_733612212 小时前
C++20概念(Concepts)入门指南
开发语言·c++·算法
AI_搬运工2 小时前
从智能指针窥见现代C++的生存法则:告别内存泄漏,这篇就够了
c++