【动态规划】回文串问题

一、回文子串

题目描述:

思路讲解:

本道题需要我们判断字符串中所有可能的子串是否为回文,并统计总数,回文子串是正着读与倒着读一致的连续子串,我们可以通过过二维状态数组记录子串是否为回文,进而统计所有回文子串的数目。以下是具体思路:

  1. 状态表示:dpij 表示字符串中从索引 i 到 j(i <= j)的子串 si...j 是否为回文子串(true 表示是,false 表示否)
  2. 状态转移方程
    • 核心条件:si == sj(子串首尾字符必须相同)
    • 辅助条件(满足其一即可):
      • 若 i == j(子串长度为 1,单个字符本身是回文)
      • 若 i + 1 == j(子串长度为 2,两个相同字符构成回文)
      • 若 i + 1 < j(子串长度>=3),则需内部子串 si+1...j-1 是回文(即 dpi+1j-1 == true)
  3. 初始化
    • 初始时 dp 数组全为 false,通过后续遍历更新状态
  4. 填写 DP 表与结果保存
    • 按照从下到上、从右到左的计算顺序:外层循环从字符串末尾索引 i = n-1 遍历到 0,内层循环从 i 遍历到 n-1(确保 i <= j),这种顺序保证计算 dpij 时,内部子串 dpi+1j-1 已被计算
    • 维护变量 ret,每当 dpij 被判定为 true(即子串 si...j 是回文)时,ret 加 1
  5. 结果返回:遍历结束后,ret 即为所有回文子串的总数

编写代码:

cpp 复制代码
class Solution {
public:
    int countSubstrings(string s) {
        int n = s.size();
        vector<vector<bool>> dp(n,vector<bool>(n,false));

        int ret = 0;
        for(int i = n - 1 ; i >= 0 ; i--)
        {
            for(int j = i ; j < n ; j++)
            {
                if(s[i] == s[j] && ( i == j || i + 1 == j || i + 1 < j && dp[i+1][j-1] == true))
                {
                    dp[i][j] = true;
                    ret++;
                }
            }
        }

        return ret;
    }
};

二、最长回文子串

题目描述:

思路讲解:

本道题需要我们判断所有可能子串是否为回文,并记录最长的那个,回文子串是正着读与倒着读一致的连续子串,我们可以通过过二维状态数组记录子串是否为回文,同时记录最长回文子串的长度和起始位置,进而获得最长回文子串。以下是具体思路:

  1. 状态表示:dpij 表示子串 si...j(i <= j)是否为回文子串(true 表示是,false 表示否)
  2. 状态转移方程
    • 核心条件:si == sj(首尾字符必须相同)
    • 辅助条件(满足其一):
      • i == j(长度为 1 的子串,单个字符必为回文)
      • i + 1 == j(长度为 2 的子串,两相同字符构成回文)
      • i + 1 < j(长度>=3),则内部子串 si+1...j-1 必须为回文(即 dpi+1j-1 == true)
  3. 初始化
    • dp 数组初始全为 false,通过遍历更新
  4. 填写 DP 表
    • 按照从下到上、从左到右的计算顺序:外层循环从字符串末尾索引 i = n-1 遍历到 0,内层循环从 i 遍历到 n-1(确保 i <= j),这种顺序保证计算 dpij 时,内部子串 dpi+1j-1 已被计算
  5. 最长回文子串追踪
    • 用 pos 记录最长回文子串的起始索引,len 记录其长度(初始为 1,默认单个字符)
    • 每当 dpij 为 true 时,计算子串长度 j - i + 1,若大于当前 len,则更新 pos = i 和 len = j - i + 1
  6. 结果返回
    • 遍历结束后,通过 s.substr(pos, len) 截取并返回最长回文子串

编写代码:

cpp 复制代码
class Solution {
public:
    string longestPalindrome(string s) {
        int n = s.size();
        vector<vector<bool>> dp(n,vector<bool>(n,false));
        int pos = n - 1 , len = 1;

        for(int i = n - 1 ; i >= 0 ; i--)
        {
            for(int j = i ; j < n ; j++)
            {
                if(s[i] == s[j] && ( i == j || i + 1 == j || i + 1 < j && dp[i+1][j-1] == true))
                {
                    dp[i][j] = true;
                    if(len < j - i + 1)
                    {
                        pos = i;
                        len = j - i + 1;
                    }
                }
            }
        }

        return s.substr(pos,len);
    }
};

三、分割回文串 IV

题目描述:

思路讲解:

本道题需要我们判断字符串能否分割为 3 个非空回文子串,关键是先高效判断任意子串是否为回文,再枚举分割点验证条件。以下是具体思路:

  1. 状态表示:dpij 表示子串 si...j(i <= j)是否为回文子串(true 表示是,false 表示否)
  2. 状态转移方程
    • 核心条件:si == sj(首尾字符必须相同)
    • 辅助条件(满足其一):
      • i == j(长度为 1 的子串,单个字符必为回文)
      • i + 1 == j(长度为 2 的子串,两相同字符构成回文)
      • i + 1 < j(长度>=3),则内部子串 si+1...j-1 必须为回文(即 dpi+1j-1 == true)
  3. 初始化
    • dp 数组初始全为 false,通过遍历更新
  4. 填写 DP 表
    • 按照从下到上、从左到右的计算顺序:外层循环从字符串末尾索引 i = n-1 遍历到 0,内层循环从 i 遍历到 n-1(确保 i <= j),这种顺序保证计算 dpij 时,内部子串 dpi+1j-1 已被计算
  5. 分割点枚举与验证
    • 定义两个分割点 i 和 j(1 <= i <= j <= n-2),将字符串分为三部分:s0...i-1、si...j、sj+1...n-1
    • 遍历所有可能的 i 和 j,若三部分子串对应的 dp0i-1、dpij、dpj+1n-1 均为 true,则返回 true
  6. 结果返回
    • 若遍历所有分割点均不满足条件,返回 false;否则返回 true

编写代码:

cpp 复制代码
class Solution {
public:
    bool checkPartitioning(string s) {
        int n = s.size();
        vector<vector<bool>> dp(n,vector<bool>(n,false));

        for(int i = n - 1 ; i >= 0 ; i--)
        {
            for(int j = i ; j < n ; j++)
            {
                if(s[i] == s[j])
                {
                    dp[i][j] = i + 1 < j ? dp[i+1][j-1] : true;
                }
            }
        }

        for(int i = 1 ; i < n - 1; i++)
        {
            for(int j = i ; j < n - 1 ; j++)
            {
                if(dp[0][i-1] && dp[i][j] && dp[j+1][n-1])
                {
                    return true;
                }
            }
        }

        return false;
    }
};

四、分割回文串 II

题目描述:

思路讲解:

本道题需要我们计算将字符串分割成一些回文串的最少分割次数,我们可以先预处理回文子串判断,再计算最少分割次数。以下是具体思路:

  1. 状态表示
    • gij 表示子串 si...j(i <= j)是否为回文子串(true 表示是)
    • fi 表示将子串 s0...i 分割为回文子串的最少分割次数
  2. 回文判断的状态转移方程(g数组)
    • 核心条件:si == sj(首尾字符相同)
    • 辅助条件(满足其一):
      • i == j(长度为 1 的子串,单个字符必为回文)
      • i + 1 == j(长度为 2 的子串,两相同字符构成回文)
      • i + 1 < j(长度>=3),则内部子串 si+1...j-1 必须为回文(即 dpi+1j-1 == true)
  3. 回文判断的填写 DP 表(g数组)
    • 按照从下到上、从左到右的计算顺序:外层循环从字符串末尾索引 i = n-1 遍历到 0,内层循环从 i 遍历到 n-1(确保 i <= j),这种顺序保证计算 dpij 时,内部子串 dpi+1j-1 已被计算
  4. 最少分割次数的状态转移(f 数组)
    • 若 g0i == true(整个子串是回文),则 fi = 0(无需分割)
    • 否则,枚举分割点 j(1 <= j <= i):若 gji == true(子串 sj...i 是回文),则 fi = min(fi, fj-1 + 1)(在 s0...j-1 的最少分割次数基础上 + 1)
  5. 初始化
    • f 数组初始化为 INT_MAX
  6. 回文判断的填写 DP 表(f数组)
    • 外层遍历 i 从 0 到 n-1,对每个 i 按上述规则更新 fi
  7. 结果返回
    • 最终结果为 fn-1(子串 s0...n-1 的最少分割次数)

编写代码:

cpp 复制代码
class Solution {
public:
    int minCut(string s) {
        int n = s.size();
        vector<int> f(n,INT_MAX);
        vector<vector<bool>> g(n,vector<bool>(n,false));

        for(int i = n - 1 ; i >= 0 ; i--)
        {
            for(int j = i ; j < n ; j++)
            {
                if(s[i] == s[j])
                    g[i][j] = i + 1 < j ? g[i+1][j-1] : true;
            }
        }

        for(int i = 0 ; i < n ; i++)
        {
            if(g[0][i] == true)
            {
                f[i] = 0;
            }
            else
            {
                for(int j = i ; j > 0 ; j--)
                {
                    if(g[j][i] == true)
                    {
                        f[i] = min(f[i],f[j-1] + 1);
                    }
                }
            }
        }

        return f[n-1];
    }
};

五、最长回文子序列

题目描述:

思路讲解:

本道题需要我们计算最长回文子序列的长度,我们可以通过二维状态数组记录子串中最长回文子序列的长度。以下是具体思路:

  1. 状态表示:dpij 表示子串 si...j(i <= j)中最长回文子序列的长度
  2. 状态转移方程
    • 若 si != sj:最长回文子序列只能来自左侧子串或右侧子串,即 dpij = max(dpi+1j, dpij-1)
    • 若 si == sj
      • 若 i + 1 < j(子串长度>=3),则 dpij = dpi+1j-1 + 2(中间子串的最长回文长度加 2,即两端字符)
      • 若 i == j(长度 1),则 dpij = 1(单个字符本身)
      • 若 i + 1 == j(长度 2),则 dpij = 2(两个相同字符)
  3. 初始化
    • 初始时 dp 数组全为 0,通过遍历更新
  4. 填写 DP 表
    • 按照从下到上,从左到右遍历:外层循环 i 从 n-1 到 0,内层循环 j 从 i 到 n-1,此顺序确保计算 dpij 时,dpi+1j、dpij-1 和 dpi+1j-1 已确定
  5. 结果返回
    • 整个字符串的最长回文子序列长度存储在 dp0n-1 中,直接返回即可

编写代码:

cpp 复制代码
class Solution {
public:
    int longestPalindromeSubseq(string s) {
        int n = s.size();
        vector<vector<int>> dp(n,vector<int>(n,0));

        for(int i = n - 1 ; i >= 0 ; i--)
        {
            for(int j = i ; j < n ; j++)
            {
                if(s[i] != s[j]) dp[i][j] = max(dp[i+1][j],dp[i][j-1]);
                else
                {
                    dp[i][j] = i + 1 < j ? dp[i+1][j-1] + 2 : j - i + 1;
                }
            }
        }

        return dp[0][n-1];
    }
};

六、让字符串成为回文串的最少插入次数

题目描述:

思路讲解:

本道题需要我们计算通过最少插入字符操作使字符串成为回文串,我们可以通过二维状态数组计算将子串转换为回文串所需的最少插入次数。以下是具体思路:

  1. 状态表示:dpij 表示将子串 si...j(i <= j)转换为回文串所需的最少插入次数
  2. 状态转移方程
    • 若 si != sj:需通过插入字符使首尾对称,选择插入次数更少的方案:
      • dpij = min(dpi+1j, dpij-1) + 1
      • dpi+1j 是忽略 si 后子串的插入次数(需插入 si 对称字符),dpij-1 是忽略 sj 后子串的插入次数(需插入 sj 对称字符),两者取最小值后加 1 次插入
    • 若 si == sj:首尾已对称,插入次数取决于内部子串:
      • dpij = i + 1 < j ? dpi+1j-1 : 0
      • 若子串长度>=3(i+1 < j),则依赖内部子串 si+1...j-1 的插入次数;若长度为 1(i == j)或 2(i+1= j),则无需插入(本身对称或已满足回文)
  3. 初始化
    • 初始时 dp 数组全为 0,通过遍历更新
  4. 填写 DP 表
    • 按照从下到上,从左到右的顺序:外层循环 i 从 n-1 到 0,内层循环 j 从 i 到 n-1,此顺序确保计算 dpij 时,dpi+1j、dpij-1 和 dpi+1j-1 已完成计算
  5. 结果返回
    • 整个字符串 s0...n-1 转换为回文串的最少插入次数存储在 dp0n-1 中,直接返回即可

本道题还可以将问题进行转换

  • 字符串要成为回文串,需保证所有字符对称。而最长回文子序列是原字符串中 "天然对称" 的最大部分,无需额外插入字符即可维持对称结构。因此:原字符串中不属于最长回文子序列的字符,都需要通过插入对称字符来补全回文结构。
  • 最少插入次数 = 字符串长度 - 最长回文子序列的长度

编写代码:

cpp 复制代码
class Solution {
public:
    int minInsertions(string s) {
        int n = s.size();
        vector<vector<int>> dp(n,vector<int>(n,0));

        for(int i = n - 1 ; i >= 0 ; i--)
        {
            for(int j = i ; j < n ; j++)
            {
                if(s[i] != s[j]) dp[i][j] = min(dp[i+1][j],dp[i][j-1]) + 1;
                else
                {
                    dp[i][j] = i + 1 < j ? dp[i+1][j-1] : 0;
                }
            }
        }

        return dp[0][n-1];
    }
};

结尾

如果有什么建议和疑问,或是有什么错误,大家可以在评论区中提出。

希望大家以后也能和我一起进步!!🌹🌹

如果这篇文章对你有用的话,希望大家给一个三连支持一下!!🌹🌹

相关推荐
变量未定义~10 小时前
既约分数、阶乘约数、逆元、最大质因子个数【算法赛】
算法
KaMeidebaby10 小时前
卡梅德生物技术快报|Western Blot 实验应用:肺肠轴机制研究全流程技术解析
前端·数据库·人工智能·算法·百度
AhriProGramming11 小时前
计算机科普故事会-<2>见微知著
算法
BD4SXV11 小时前
线性二次调节器(Linear Quadratic Regulator,LQR)的无限时域最优控制求解与黎卡提方程
算法·自动化
ST——Jess11 小时前
2026年度传统文化数字化与命理科技(Ethno-tech)行业趋势研究报告:专业级数智工作台的技术壁垒与评测标准
人工智能·科技·算法·架构
Matrix_1112 小时前
第13篇:非线性位移场——漩涡、鱼眼、水波纹与球面化
图像处理·算法
金牌归来发现妻女流落街头12 小时前
【LeetCode 第207题】
算法·leetcode·拓扑·领接表
熬夜敲代码的猫12 小时前
AVL树(C++详解版)
数据结构·c++·算法
-To be number.wan12 小时前
算法日记 | STL-MAP
c++·算法
cjp56012 小时前
015. UG 二次开发,拉伸草图生成实体类,高级草图类封装
算法