【动态规划刷题 18】(hard)回文子串&& (hard)最长回文子串

简介: 【动态规划刷题 18】(hard)回文子串&& (hard)最长回文子串

1745. 分割回文串 IV

给你一个字符串 s ,如果可以将它分割成三个 非空 回文子字符串,那么返回 true ,否则返回 false 。

当一个字符串正着读和反着读是一模一样的,就称其为 回文字符串 。

示例 1:

输入:s = “abcbdd”

输出:true

解释:“abcbdd” = “a” + “bcb” + “dd”,三个子字符串都是回文的。

示例 2:

输入:s = “bcbddxy”

输出:false

解释:s 没办法被分割成 3 个回文子字符串。

解法:

算法思路:

题⽬要求⼀个字符串被分成「三个⾮空回⽂⼦串」,乍⼀看,要表⽰的状态很多,有些⽆从下⼿。

其实,我们可以把它拆成「两个⼩问题」:

i. 动态规划求解字符串中的⼀段⾮空⼦串是否是回⽂串;

ii. 枚举三个⼦串除字符串端点外的起⽌点,查询这三段⾮空⼦串是否是回⽂串。

那么这道困难题就免秒变为简单题啦,变成了⼀道枚举题

代码:

  bool checkPartitioning(string s) {
           int n=s.size();
        vector<vector<bool>> dp(n,vector<bool>(n));
        dp[0][0]=1;
        for(int j=1;j<n;j++)
        {
            dp[j][j]=1;
            for(int i=0;i<=j;i++)
            {
                if(s[i]==s[j])
                {
                    if(j==i+1) dp[i][j]=1;
                    if(j>i+1)  dp[i][j]=dp[i+1][j-1];
                }
            }
        }
        for(int i=1;i<n-1;i++)
        {
            for(int j=0;j<i;j++)
            {
                if(dp[0][j]&&dp[j+1][i]&&dp[i+1][n-1])
                return true;
            }
        }
        return false;
    }




925f331a7de941a7816b0e1d2df5356d.png

132. 分割回文串 II

给你一个字符串 s,请你将 s 分割成一些子串,使每个子串都是回文。

返回符合要求的 最少分割次数 。

示例 1:

输入:s = “aab”

输出:1

解释:只需一次分割就可将 s 分割成 [“aa”,“b”] 这样两个回文子串。

示例 2:

输入:s = “a”

输出:0

示例 3:

输入:s = “ab”

输出:1

算法思路:状态表⽰:

根据「经验」,继续尝试⽤ i 位置为结尾,定义状态表⽰,看看能否解决问题:

dp[i] 表⽰: s 中 [0, i] 区间上的字符串,最少分割的次数

1.状态表示*

根据「经验」,继续尝试⽤ i 位置为结尾,定义状态表⽰,看看能否解决问题:

dp[i] 表⽰: s 中 [0, i] 区间上的字符串,最少分割的次数⽂串。2.状态转移方程

状态转移⽅程⼀般都是根据「最后⼀个位置」的信息来分析:设 0 <= j <= i ,那么我们可以根据 j ~ i 位置上的⼦串是否是回⽂串分成下⾯两类:

i. 当 [j ,i] 位置上的⼦串能够构成⼀个回⽂串,那么 dp[i] 就等于 [0, j - 1] 区 间上最少回⽂串的个数+1,即 dp[i] = dp[j - 1] + 1 ;

ii. 当 [j ,i] 位置上的⼦串不能构成⼀个回⽂串,此时 j 位置就不⽤考虑。

3. 初始化

观察「状态转移⽅程」,我们会⽤到 j - 1 位置的值。我们可以思考⼀下当 j == 0 的时候,表⽰的区间就是 [0, i] 。如果 [0, i] 区间上的字符串已经是回⽂串了,最⼩的回⽂串就是 了, j 往后的值就不⽤遍历了。

因此,我们可以在循环遍历 j 的值之前处理 j == 0 的情况,然后 j 从 1 开始循环。

但是,为了防⽌求 min 操作时, 0 ⼲扰结果。我们先把表⾥⾯的值初始化为「⽆穷⼤」

4. 填表顺序

从左往右

5. 返回值

根据「状态表⽰」,应该返回 dp[n - 1]

代码:

 int minCut(string s) {
  int n=s.size();
        vector<vector<bool>> dp(n,vector<bool>(n));
        dp[0][0]=1;
        for(int j=1;j<n;j++)
        {
            dp[j][j]=1;
            for(int i=0;i<=j;i++)
            {
                if(s[i]==s[j])
                {
                    if(j==i+1)  dp[i][j]=1;
                    if(j>i+1)   dp[i][j]=dp[i+1][j-1];
                }
            }
        }
        vector<int> dp1(n,INT_MAX-500);
        dp1[0]=0;
        for(int i=1;i<n;i++)
        {
            if(dp[0][i]) dp1[i]=0;
            else
            {
                for(int j=1;j<=i;j++)
                {
                    if(dp[j][i])
                        dp1[i]=min(dp1[i],dp1[j-1]+1);
                }
            }
        }
        return dp1[n-1];
    }

fe7eaf3caead4172840b63c146c91238.png


相关文章
|
7月前
【洛谷 P1219】[USACO1.5]八皇后 Checker Challenge 题解(深度优先搜索+回溯法)
**USACO1.5八皇后挑战**是关于在$n\times n$棋盘上放置棋子的,确保每行、每列及两条主对角线上各有一个且仅有一个棋子。给定$6$作为输入,输出前$3$个解及解的总数。例如,对于$6\times6$棋盘,正确输出应包括解的序列和总数。代码使用DFS解决,通过跟踪对角线占用状态避免冲突。当找到所有解时,显示前三个并计数。样例输入$6$产生输出为解的前三个排列和总数$4$。
44 0
|
8月前
|
索引
力扣---最长回文子串(动态规划)
力扣---最长回文子串(动态规划)
|
8月前
[leetcode~数位动态规划] 2719. 统计整数数目 hard
[leetcode~数位动态规划] 2719. 统计整数数目 hard
|
8月前
|
算法
【面试算法——动态规划 21】不同的子序列(hard)&& 通配符匹配(hard)
【面试算法——动态规划 21】不同的子序列(hard)&& 通配符匹配(hard)
106 1
|
8月前
|
算法
【面试算法——动态规划 19】最长回文子序列&& (hard)让字符串成为回文串的最少插入次数
【面试算法——动态规划 19】最长回文子序列&& (hard)让字符串成为回文串的最少插入次数
|
8月前
【每日一题Day122】LC1237找出给定方程的正整数解 | 双指针 二分查找
【每日一题Day122】LC1237找出给定方程的正整数解 | 双指针 二分查找
48 0
|
8月前
|
机器学习/深度学习 算法
【动态规划刷题 17】回文子串&& 最长回文子串
【动态规划刷题 17】回文子串&& 最长回文子串
【动态规划刷题】整数拆分
【动态规划刷题】整数拆分
98 0
|
Java 测试技术
hdu1231 最大连续子序列【动态规划】
hdu1231 最大连续子序列【动态规划】
44 0
|
存储 人工智能 算法
剑指offer(C++)-JZ42:连续子数组的最大和(算法-动态规划)
剑指offer(C++)-JZ42:连续子数组的最大和(算法-动态规划)