Longest Palindromic Substring最长回文子串-学习动态规划Dynamic Programming(DP)

通过leetcode的查找Longest Palindromic Substring最长回文子串,来学习一下动态规划。

动态规划的核心是能把当前问题分解成已经有的更小的问题,然后直到这个问题有一个常数答案,并且这个分解过程有一个固定模式。

所以最终目的就是寻找 1. 固定的分解模式 2.常数答案。

根据回文串的定义可知,去掉回文串的首位,剩下的依然是回文串,重复这个过程,我们可以得到最终只有长度为1(奇数长度)或者0(偶数长度)的字符串,也就是一个常量解。

因此我们就得到了

  1. 固定的分解模式,及回文串去掉首尾依然是回文串,翻过来,回文串加上相同的首尾依然是回文串。
  2. 常数答案,2.1 一个字符是回文串, 2.2 两个相邻字符如果相同是回文串

假设 s为字符串,i和j表示索引,dpij=tue/false 分别表示字符串si...j是/否为回文

则 dpij=dpi+1j-1 && si==sj,及当si+1...j-1是回文且si=sj时,那么si...j也是回文,及dpij=true;

因此可以得到转换方程

java 复制代码
if (s.charAt(i) == s.charAt(j)) {
      if (j - i <= 2) { //j=i||j=i+1
         dp[i][j] = true;
      } else {
         dp[i][j] = dp[i + 1][j - 1];
      }
} else {
      dp[i][j] = false;
}

实践中有两种写法,一种是从尾向前计算,好处是当在 (i,j) 时,(i+1,j-1) 一定已经被计算,另外一种是从头向后计算,好处是比较直观。

从尾向前:

java 复制代码
public String longestPalindrome(String s) {
        int n = s.length();
        if (n < 2) {
            return s;
        }
        int maxLen = 0;
        int maxIndex = 0;
        boolean[][] dp = new boolean[n][n];

        for (int i = n - 1; i >= 0; i--) {
            for (int j = i; j < n; j++) {
                if (s.charAt(i) == s.charAt(j)) {
                    if (j - i <= 2) {
                        dp[i][j] = true;
                    } else {
                        dp[i][j] = dp[i + 1][j - 1];
                    }
                    if (dp[i][j] && (j - i + 1 > maxLen)) {
                        maxLen = j - i + 1;
                        maxIndex = i;
                    }
                } else {
                    dp[i][j] = false;
                }
            }
        }
        return s.substring(maxIndex, maxIndex + maxLen);
    }

从头向后

java 复制代码
public String longestPalindrome(String s) {
        int n = s.length();
        if (n < 2) {
            return s;
        }
        int maxLen = 0;
        int maxIndex = 0;
        boolean[][] dp = new boolean[n][n];

        for (int len = 1; len <= n; len++) {
            for (int i = 0; i + len - 1 < n; i++) {
                int j = i + len - 1;
                if (s.charAt(i) == s.charAt(j)) {
                    if (j - i <= 2) {
                        dp[i][j] = true;
                    } else {
                        dp[i][j] = dp[i + 1][j - 1];
                    }
                    if (dp[i][j] && (j - i + 1 > maxLen)) {
                        maxLen = j - i + 1;
                        maxIndex = i;
                    }
                } else {
                    dp[i][j] = false;
                }
            }
        }
        return s.substring(maxIndex, maxIndex + maxLen);
    }
相关推荐
vibecoding日记16 小时前
双非如何快速入职字节等大厂大模型?真实案例分析:推理优化和投机解码
算法·求职·大模型工程师
yszaygr213818 小时前
Verilog参数化游程编码RLE模块
算法
望易19 小时前
刚设计的大模型架构-双域耦合认知框架
算法·架构
复杂网络1 天前
多个 Claude Code 与多个 Codex 协同工作:设计与实现方案
算法
HjhIron2 天前
面试常客:字符串算法从入门到进阶
算法·面试
吴佳浩2 天前
DeepSeek DSpark:Confidence-Scheduled Speculative Decoding 技术解析
人工智能·算法·deepseek
触底反弹2 天前
🧠 搞懂 Token,才算真正入门大模型——从分词原理到 Embedding 语义实战
javascript·人工智能·算法
vivo互联网技术2 天前
ICLR 2026 | 基于后验采样的图像恢复方法LearnIR:人脸去阴影、去雾
人工智能·算法·aigc
浮生望2 天前
JS字符串与回文算法:从包装类到双指针的面试进阶之路
javascript·算法