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 维护当前最右回文边界信息
相关推荐
JieE21219 小时前
LeetCode 56. 合并区间|超清晰 JS 图解思路,面试高频区间题
javascript·算法·面试
Jack201 天前
HarmonyOS开发中错误处理策略:网络异常统一处理
算法
小小杨树1 天前
读懂色彩:拍照调色不再难
算法·计算机视觉·配色
JieE2122 天前
LeetCode 226. 翻转二叉树|JS 递归超详细拆解,二叉树入门经典题
javascript·算法
JieE2122 天前
LeetCode 104. 二叉树的最大深度|递归思路超详细拆解
javascript·算法
vivo互联网技术2 天前
CVPR 2026 | 全新强化学习框架 BeautyGRPO:重塑真实人像
算法·大模型·cvpr·影像
Darling噜啦啦2 天前
列表转树算法深度解析:从 Map 到 Reduce 的两种实现,面试高频考点
数据结构·算法·面试
clint4562 天前
C++进阶(1)——前景提要
c++
用户497863050732 天前
(一)小红的数组操作
算法·编程语言