【Scramble String】cpp
题目:
Given a string s1, we may represent it as a binary tree by partitioning it to two non-empty substrings recursively.
Below is one possible representation of s1 = "great":
great
/ \
gr eat
/ \ / \
g r e at
/ \
a t
To scramble the string, we may choose any non-leaf node and swap its two children.
For example, if we choose the node "gr" and swap its two children, it produces a scrambled string "rgeat".
rgeat
/ \
rg eat
/ \ / \
r g e at
/ \
a t
We say that "rgeat" is a scrambled string of "great".
Similarly, if we continue to swap the children of nodes "eat" and "at", it produces a scrambled string "rgtae".
rgtae
/ \
rg tae
/ \ / \
r g ta e
/ \
t a
We say that "rgtae" is a scrambled string of "great".
Given two strings s1 and s2 of the same length, determine if s2 is a scrambled string of s1.
代码:
class Solution {
public:
bool isScramble(string s1, string s2) {
const int n1 = s1.size();
const int n2 = s2.size();
if (n1!=n2) { return false; }
const int n = n1;
int alpha[] = {};
for ( int i=; i<n; ++i ){ alpha[s1[i]-'a']++; alpha[s2[i]-'a']--; }
for ( int i=; i<; ++i ){ if ( alpha[i]!= ) return false; }
// terminal condition
if ( n== ) return s1[]==s2[];
// recursive process
for ( int i=; i<n; ++i ){
//cout << s1 << "," << s2 << ":" << i << endl;
if (
(
Solution::isScramble(s1.substr(,i), s2.substr(,i)) &&
Solution::isScramble(s1.substr(i,n-i), s2.substr(i,n-i))
)
||
(
Solution::isScramble(s1.substr(,i), s2.substr(n-i,i)) &&
Solution::isScramble(s1.substr(i,n-i), s2.substr(,n-i))
)
)
{ return true; }
}
return false;
}
};
tips:
这道题的题意自己并没有理解好,引用一个网上其他人的理解如下:
(http://www.blogjava.net/sandy/archive/2013/05/22/399605.html)
由于一个字符串有很多种二叉表示法,貌似很难判断两个字符串是否可以做这样的变换。
“对付复杂问题的方法是从简单的特例来思考,从而找出规律。
先考察简单情况:
字符串长度为1:很明显,两个字符串必须完全相同才可以。
字符串长度为2:当s1="ab", s2只有"ab"或者"ba"才可以。
对于任意长度的字符串,我们可以把字符串s1分为a1,b1两个部分,s2分为a2,b2两个部分,满足((a1~a2) && (b1~b2))或者 ((a1~b2) && (a1~b2))”
理解了题意,代码也就写出来了。
具体还有几个细节需要注意:
1. 为了剪枝并加快速度,做了如下几件事情:
a) 判断s1与s2的长度是否相等
b) 判断s1与s2的每个字符数量是否相等(这里由于是字母所以用一个定长数组alpha[26]表示:某个字母在s1中出现一次+1,在s2中出现一次-1;最终alpha的每个元素都是0则证明s1与s2的每个字符数量相等。扩展一下,如果字符不止26个字母,包含其他字符呢?可以用hashmap表示)
2. 设定终止条件:
如果s1和s2长度已经为1,无法再分割了,就直接比较即可。
3. 在递归传入参数的时候,用到了substr(begin, num):
a) begin代表切取的第一个字符下标,num代表截取几个字符
b) 注意每次传入isScramble的字符长度相等
===========================================
上述的做法类似记忆化搜索,网上还有一种动态规划的解法,也学习了吧。
(http://blog.csdn.net/linhuanmars/article/details/24506703)
class Solution {
public:
bool isScramble(string s1, string s2) {
const int n1 = s1.size();
const int n2 = s2.size();
if ( n1 != n2 ) return false;
const int n = n1;
vector<vector<vector<bool> > > dp(n,vector<vector<bool> >(n,vector<bool>(n+,false)));
for ( int k=; k<=n; ++k )
{
for ( int i=; i<=n-k; ++i )
{
for ( int j=; j<=n-k; ++j )
{
if ( k== )
{
dp[i][j][k] = s1[i]==s2[j];
continue;
}
for ( int l=; l<k; ++l )
{
dp[i][j][k] =
(dp[i][j][l] && dp[i+l][j+l][k-l])
||
(dp[i][j+k-l][l] && dp[i+l][j][k-l]);
if ( dp[i][j][k] ) break;
}
}
}
}
/*
for ( int k=0; k<=n; ++k )
{
cout << k << endl;
for ( int i=0; i<n; ++i)
{
for (int j=0; j<n; ++j )
{
cout << dp[i][j][k] << " ";
}
cout << endl;
}
}
*/
return dp[][][n];
}
};
tips:
AC之后发现这道题的dp思路其实可以由递归思路得来。
递归算法在不断的递归过程中,其实是一直再算s1的某一段与s2等长的某一段是否符合scramble的特点;注意,这里的某一段不一定指的是s1和s2从同一个位置开始。递归过程中,并没有记录这样的s1、s2字串比较的历史信息;而dp的解法是比较一次记录一次比较的历史信息,下次再判断的时候就可以利用上历史的比较信息了。
dp的过程(http://blog.csdn.net/linhuanmars/article/details/24506703)已经说的很好了。
这里有个细节需要注意一下,就是最外层的循环k代表从s1和s2截取字符串的长度。这里为了在下标表示方便,定义为n+1维;这样的好处就在于循环中的k直接表示的就是需要比较的子字符串的长度,不用考虑k-1这一类的内容。
这题的dp思路太精妙,只能学习膜拜。
完毕。
===================================================
第二次过这道题,dp的做法没时间去过了,用“深搜+剪枝”的做法更直观一些。
class Solution {
public:
bool isScramble(string s1, string s2) {
if ( s1.size()!=s2.size() ) return false;
int count[] = {};
for ( int i=; i<s1.size(); ++i ){
count[(int)s1[i]]++;
count[(int)s2[i]]--;
}
for ( int i=; i<; ++i ) { if ( count[i]!= ) return false; }
if ( s1.size()== ) return s1[]==s2[];
for ( int l=; l<s1.size(); ++l ){
bool possible = Solution::isScramble(s1.substr(,l), s2.substr(,l)) &&
Solution::isScramble(s1.substr(l, s1.size()-l), s2.substr(l, s2.size()-l));
if ( possible ) return true;
possible = Solution::isScramble(s1.substr(,l), s2.substr(s2.size()-l,l)) &&
Solution::isScramble(s1.substr(l,s1.size()-l), s2.substr(,s2.size()-l));
if ( possible ) return true;
}
return false;
}
};
【Scramble String】cpp的更多相关文章
- 【Interleaving String】cpp
题目: Given s1, s2, s3, find whether s3 is formed by the interleaving of s1 and s2. For example,Given: ...
- 【Valid Sudoku】cpp
题目: Determine if a Sudoku is valid, according to: Sudoku Puzzles - The Rules. The Sudoku board could ...
- 【WildCard Matching】cpp
题目: Implement wildcard pattern matching with support for '?' and '*'. '?' Matches any single charact ...
- 【Add binary】cpp
题目: Given two binary strings, return their sum (also a binary string). For example,a = "11" ...
- 【Implement strStr() 】cpp
题目: Implement strStr(). Returns the index of the first occurrence of needle in haystack, or -1 if ne ...
- 【Valid Palindrome】cpp
题目: Given a string, determine if it is a palindrome, considering only alphanumeric characters and ig ...
- 【Valid Parentheses】cpp
题目: Given a string containing just the characters '(', ')', '{', '}', '[' and ']', determine if the ...
- 【Simplify Path】cpp
题目: Given an absolute path for a file (Unix-style), simplify it. For example,path = "/home/&quo ...
- 【Valid Number】cpp
题目: Validate if a given string is numeric. Some examples:"0" => true" 0.1 " = ...
随机推荐
- 《Python高效开发实战》实战演练——开发Django站点1
6.2 实战演练:开发Django站点 用Django开发网站需要遵循Django的一套开发流程.本节通过建立一个消息录入页面演示Django的开发流程及相关技术. 6.12.1 建立项目 在进行D ...
- JIRA Plugin Development——Configurable Custom Field Plugin
关于JIRA Plugin开发的中文资料相当少,这可能还是由于JIRA Plugin开发在国内比较小众的原因吧,下面介绍下自己的一个JIRA Plugin开发的详细过程. 业务需求 创建JIRA IS ...
- RobotFramework:钉钉扫码登录UI自动化
背景: 遇到一个项目,使用的是钉钉扫码登录,一时间不知道该怎么下手了,还是先F12抓包看下都有什么数据传输吧. 分析: 先熟悉下钉钉扫码登录的逻辑,参考官文:https://open-doc.ding ...
- 在C++类中使用dllimport和dllexport导出,
在Windows平台下: 您可以使用dllimport或dllexport属性声明C ++类.这些形式意味着导入或导出整个类.以这种方式导出的类称为可导出类. 以下示例定义可导出的类.导出其所有成员函 ...
- Python监控日志中经常访问的ip
一.需求:每分钟检查一次日志文件,如果这一分钟内同一个ip请求次数超过200次,加入黑名单 1.日志文件中,每一行的格式为:XXX.XXX.XXX.XXX - - [04/Jun/2017:05:25 ...
- modprobe与insmod的区别
linux设备驱动有两种加载方式insmod和modprobe,下面谈谈它们用法上的区别1.insmod一次只能加载特定的一个设备驱动,且需要驱动的具体地址.写法为: insmod dr ...
- OO第13-14次作业总结
目录 面向对象第13-14次作业总结博客 1.设计分析 2.架构总结.测试 3.课程收获和建议 面向对象第13-14次作业总结博客 1.设计分析 这个单元是我做的最差的一个单元.总工程量超过2000行 ...
- java算法面试题:递归算法题1
递归算法题1 一个整数,大于0,不用循环和本地变量,按照n,2n,4n,8n的顺序递增,当值大于5000时,把值按照指定顺序输出来.例:n=1237则输出为:1237,2474,4948,9896,9 ...
- jqweui 中的tabbar导航
最近做微信的服务号项目,用的jqweui作为主要的ui,但是对于用惯了ele ui的开发者来说,文档貌似有点不友好.真是很让人头疼! 所以结合着自己做的项目,随便写一点东西. 比如说,tabbar导航 ...
- CSS清除浮动方法总结
什么是CSS清除浮动? 在非IE浏览器(如Firefox)下,当容器的高度为auto,且容器的内容中有浮动(float为left或right)的元素,在这种情况下,容器的高度不能自动伸长以适应内容的高 ...