【palindrome partitioning II】cpp
题目:
Given a string s, partition s such that every substring of the partition is a palindrome.
Return the minimum cuts needed for a palindrome partitioning of s.
For example, given s = "aab",
Return 1 since the palindrome partitioning ["aa","b"] could be produced using 1 cut.
代码:
class Solution {
public:
int minCut(string s)
{
const int len = s.size();
vector<vector<bool> > p(len, vector<bool>(len,false));
Solution::allPalindrome(p, s);
vector<int> dp(len, INT_MAX);
dp[] = ;
for ( int i = ; i < len; ++i )
{
for ( int j = i; j >= ; --j )
{
if ( j== )
{
if ( p[j-][i] )
{
dp[i] = ;
break;
}
}
if ( p[j][i] )
{
dp[i] = std::min(dp[i], dp[j-]+);
}
}
}
return dp[len-];
}
// palindromes[i][j] denotes if s[i:j] is palindrome
static void allPalindrome(vector<vector<bool> >& palindromes, string& s)
{
for ( int i = palindromes.size()-; i >= ; --i )
{
for ( int j = i; j <= palindromes.size()-; ++j )
{
if ( (i+>j- && s[i]==s[j]) || (i+<=j- && s[i]==s[j] && palindromes[i+][j-]) )
{
palindromes[i][j] = true;
}
}
}
}
};
tips:
通过此题掌握了动态规划的一些套路。
动规的核心在于求解两个重要的变量
p[i][j] : 代表s[i]到s[j]是否构成一个回文
dp[i] : 标记s[0]到s[i]的最小回文分割
接下来阐述上述两个变量的生成过程:
1. p[i][j]
正确地求出p[i][j]就是成功的一半,第一次没有AC主要就是p[i][j]求错了。在这里详细记录一下正确的求解过程
第一步,要明确p[i][j]的含义,即s[i]到s[j]是否为回文。
第二步,把求解p[i][j]转化为状态转移方程,即p[i][j] = f(p[k][v]) (k<=i, v<=j, 且k==i,v==j不同时成立)。
这个看起来是一个二维的dp方程,条件什么的比较麻烦,但实际上方程比较简单。
p[i][j] = s[i]==s[j] && p[i+1][j-1]
相当于两个条件
a) 首尾相等
b) 中间子字符串是回文
当然,还要处理一些极端情况:
极端情况1:i==j
极端情况2:i==j+1
把极端的case处理好,就可以开始dp过程了
另,因为是回文(正反都一样)所以求解p的时候,只用求解上三角阵就可以了。
2. dp[i]
之前理解dp有一个误区:认为dp[i]由dp[i-1]一步推倒过来。这是非常不正确的。
正确的理念应该是:dp[i]应该由dp[0]~dp[i-1]导出。(即当前结受到已计算出的中间结果影响,只不过有时候用不到回溯那么靠前的中间结果,只用回溯上一步的结果dp[i-1]就可以了)
然而,这道题正好不能只回溯一步,而是要回溯所有之前步骤。
既然要回溯,就要有规则,规则要保证不重不漏。这里我们关注的点是s[i]:即,我们求dp[i]则s[i]是一定要包含在其中某个回文字串中的,具体如下:
s[i]自己是一个回文子串
s[i-1:i]构成回文子串
s[i-2:i]构成回文子串
...
s[0:i]构成回文串
上述的i+1种情况如果都讨论到了,则dp[i]也就求出来了。
具体阐述如下:
s[0] ... s[i-2] s[i-1] s[i]:如果p[i-1][i]==true (即s[i-1]~s[i]是回文),则如果切割点在s[i-2]与s[i-1]之间,则dp[i] = min ( dp[i] , dp[i-2]+1)
....
s[0] ... s[j-1] s[j] ... s[i]:如果p[j][i]==true,则如果切割点在dp[i] = min ( dp[i], dp[j-1]+1)
s[0] ... s[i] :如果p[0][i]==true,则不用切割也可以,故最小切割数是0。
按照上述的思路,再处理一下corner case,代码也就出来了。
=====================================
这份代码在处理corner cases上面做的不太好,有很多可以避免的冗余。但是先保留着吧,毕竟是原始的思考思路。第二遍刷的时候,再精细化一些corner case。
======================================
第二次过这道题,看了第一次做的思路,代码重新写了一遍,比第一次过的时候要简洁一些。
class Solution {
public:
int minCut(string s)
{
const int n = s.size();
vector<vector<bool> > p(n, vector<bool>(n,false));
Solution::isPalindrome(p, s);
int dp[n];
fill_n(&dp[], n, );
for ( int i=; i<n; ++i )
{
if ( p[][i] )
{
dp[i]=;
continue;
}
int cuts = INT_MAX;
for ( int j=i; j>; --j)
{
if ( p[j][i] ) cuts = min(cuts, dp[j-]+);
if ( cuts== ) break;
}
dp[i] = cuts;
}
return dp[n-];
}
static void isPalindrome(vector<vector<bool> >& p, string s)
{
for ( int i=; i<s.size(); ++i )
{
for ( int j=; j<=i; ++j )
{
if ( i-j< )
{
p[j][i] = s[i]==s[j];
}
else
{
p[j][i] = s[i]==s[j] && p[j+][i-];
}
}
}
}
};
【palindrome partitioning II】cpp的更多相关文章
- 【Word Break II】cpp
题目: Given a string s and a dictionary of words dict, add spaces in s to construct a sentence where e ...
- 【Unique Paths II】cpp
题目: Follow up for "Unique Paths": Now consider if some obstacles are added to the grids. H ...
- 【Path Sum II】cpp
题目: Given a binary tree and a sum, find all root-to-leaf paths where each path's sum equals the give ...
- 【Spiral Matrix II】cpp
题目: Given an integer n, generate a square matrix filled with elements from 1 to n2 in spiral order. ...
- 【Jump Game II 】cpp
题目: Given an array of non-negative integers, you are initially positioned at the first index of the ...
- 【Combination Sum II 】cpp
题目: Given a collection of candidate numbers (C) and a target number (T), find all unique combination ...
- 【Word Ladder II】cpp
题目: Given two words (start and end), and a dictionary, find all shortest transformation sequence(s) ...
- 【Single Num II】cpp
题目: Given an array of integers, every element appears three times except for one. Find that single o ...
- 【leetcode】Palindrome Partitioning II
Palindrome Partitioning II Given a string s, partition s such that every substring of the partition ...
随机推荐
- U盘小偷——C++实现U盘插入检测和文件扫描拷贝
前几天女朋友说老师上课的PPT不共享,没法复习,想着写个U盘小偷拷贝PPT来着,后来觉得这样的行为这是不对的,万一不小心复制了老师的专利啥的,或者一些不可描述的东西,就闹大了. 虽然没有采取实际行动, ...
- UVA 11404 Plalidromic Subsquence (回文子序列,LCS)
最长回文子序列可以用求解原串s和反转串rv的LCS来得到,因为要求回文串分奇偶,dp[i][j]保存长度, 要求字典序最小,dp[i][j]应该表示回文子序列的端点,所以边界为单个字符,即i+j=le ...
- 【BZOJ4458】GTY的OJ(树上超级钢琴)
点此看题面 大致题意: 给你一棵树,让你求出每一个节点向上的长度在\([l,r]\)范围内的路径权值和最大的\(m\)条路径的权值总和. 关于此题的数列版本 此题的数列版本,就是比较著名的[BZOJ2 ...
- 【洛谷1993】小K的农场(差分约束系统模板题)
点此看题面 大致题意: 给你若干组不等式,请你判断它们是否有解. 差分约束系统 看到若干组不等式,应该很容易想到差分约束系统吧. \(A-B≥C\):转换可得\(A-B≥C\) \(A-B≤C\):转 ...
- Problem A: 李白打酒
Problem A: 李白打酒 Time Limit: 1 Sec Memory Limit: 128 MBSubmit: 825 Solved: 373[Submit][Status][Web ...
- css img 等比例自动缩放
按父容器宽度自动缩放,并且保持图片原本的长宽比 img{ width: auto; height: auto; max-width: 100%; max-height: 100%; }
- ASP.NET各种技巧
1.动态添加文件框 前台页面关键部分: <script type="text/javascript"> //添加一个选项 function AddFileCtrol() ...
- 题解 P2626 【斐波那契数列(升级版)】
这道题,大家一定要注意: 要对2^31取模 ! ( 本蒟蒻开始没注意到这一点,WA了 ) (不过大家在试样例的时候,试试47,出不了结果,就说明你没模2^31) 总体来说,这道题考查的知识点就两个: ...
- Repeater控件里面取不到CheckBox的值
然后在后台需要取到,选中了那些 然后在后台怎么取也取不到,当时就纳闷了,然后开始怀疑自己的代码是不是错了,经过仔细一看,我的妈呀,加载事件了差一句话......整个人都不好了 加载事件差这句话不行,补 ...
- 一次线上CPU高的问题排查实践
一次线上CPU高的问题排查实践 前言 近期某一天上班一开电脑,就收到了运维警报,有两台服务CPU负载很高,同时收到一线同事反馈 系统访问速度非常慢,几乎无响应. 一个美好的早晨,最怕什么就来什么.只好 ...