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 维护当前最右回文边界信息
相关推荐
zmzb01039 小时前
C++课后习题训练记录Day120
开发语言·c++
ximu_polaris9 小时前
设计模式(C++)-行为型模式-状态模式
c++·设计模式·状态模式
ximu_polaris9 小时前
设计模式(C++)-行为型模式-迭代器模式
c++·设计模式·迭代器模式
tjl521314_219 小时前
01C++ 类定义与访问控制(封装)
java·开发语言·c++
阿Y加油吧9 小时前
二刷 LeetCode:152. 乘积最大子数组 & 416. 分割等和子集 复盘笔记
笔记·算法·leetcode
万添裁10 小时前
huawei 机考
算法·华为·深度优先
IronMurphy17 小时前
【算法四十三】279. 完全平方数
算法
墨染天姬18 小时前
【AI】Hermes的GEPA算法
人工智能·算法
papership18 小时前
【入门级-数据结构-3、特殊树:完全二叉树的数组表示法】
数据结构·算法·链表
smj2302_7968265218 小时前
解决leetcode第3911题.移除子数组元素后第k小偶数
数据结构·python·算法·leetcode