题目miaos
给你一个字符串 s,请你将 s 分割成一些子串,使每个子串都是回文。返回符合要求的 最少分割次数 。
示例 1:
输入:s = “aab”
输出:1
解释:只需一次分割就可将 s 分割成 [“aa”,“b”] 这样两个回文子串。
示例 2:
输入:s = “a”
输出:0
示例 3:
输入:s = “ab”
输出:1
二重动态规划:
一重动态规划求解的是回文串,判断每个子串是否是回文串,
- 当i>j 时,空串也是回文串,其次当i<=j 时,
- 当s[i]==s[j] 且 f[i+1][j-1] =true 时 f[i][j] =true ,边界情况,子串长度小于等于2。
第二重动态规划是去寻找最小的分割次数。
- 状态表示:f[i] 表示把前i个字符串分割成回文串需要的最少分割次数。
- 状态计算:枚举回文串的起点i ,枚举最后一段回文串的起点 i,然后利用 st[i][j] 可知 s[i…j]是否是回文串,如果是回文串,则 f[j] 可以从 f[i−1]+1转移
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));
for(int j=0;j<n;j++)
{
for(int i=0;i<=j;i++)
{
if(i==j) g[i][j]=true;
if(s[i]==s[j] && (j-i<=2 || g[i+1][j-1])) g[i][j]=true;
}
}
for(int j=0;j<n;j++)
{
if(g[0][j]) f[j]=0;
else
{
for(int i=0;i<j;i++)
{
if(g[i+1][j]) f[j]=min(f[j],f[i]+1);
}
}
}
return f[n-1];
}
};