Leetcode - Scramble String

Screenshot from 2016-01-31 21:01:22.png

My code:

public class Solution {
    public boolean isScramble(String s1, String s2) {
        if (s1 == null || s2 == null)
            return false;
        else if (s1.length() != s2.length())
            return false;
        else if (!isSame(s1, s2))
            return false;
        else if (s1.equals(s2))
            return true;
        /** split string: [0, i), [i, s1.length()) */
        for (int i = 1; i < s1.length(); i++) {
            String s11 = s1.substring(0, i);
            String s12 = s1.substring(i, s1.length());
            String s21 = s2.substring(0, i);
            String s22 = s2.substring(i, s2.length());
            String s23 = s2.substring(s1.length() - i, s1.length());
            String s24 = s2.substring(0, s1.length() - i);
            if (isScramble(s11, s21) && isScramble(s12, s22))
                return true;
            if (isScramble(s11, s23) && isScramble(s12, s24))
                return true;
        }
        return false;
    }
    
    private boolean isSame(String s1, String s2) {
        char[] c1 = s1.toCharArray();
        char[] c2 = s2.toCharArray();
        Arrays.sort(c1);
        Arrays.sort(c2);
        s1 = new String(c1);
        s2 = new String(c2);
        return s1.equals(s2);
    }
}

这道题目我是没有思路的。看了答案之后才有了思路。
这个解法是暴力解法,具体复杂度是多少,我觉得是 2 ^ n
s1, s2, 如果从 i 处切开
那么就比较,
s1[0, i) with s2[0, i) and s1[i, len) with s2[i, len) 是否为scramble
然后这两个子串可以继续递归下去。
这是一种情况。
第二种情况,
s1[0, i) with s2[len - i, len) and s1[i, len) with s2[0, len - i) 是否为scramble
这两种情况只要有一种情况是scramble,就return true,否则return false
然后进一步递归下去。
复杂度个人估计,O(2 ^ n)

解法2,
Dynamic Programming
My code:

public class Solution {
    public boolean isScramble(String s1, String s2) {
        if (s1 == null || s2 == null)
            return false;
        else if (s1.length() != s2.length())
            return false;
        else if (!isSame(s1, s2))
            return false;
        else if (s1.equals(s2))
            return true;
        /** using dp 
         *  dp[i][j]][k] = 1 means s1.substring(i, i + k) and s2.substring(j, j + k) is scramble
         */
        int len = s1.length();
        int[][][] dp = new int[len][len][len + 1];
        /** for single character, if s1(i) == s2(j), then dp[i][j][1] = 1 */
        for (int i = 0; i < len; i++) {
            for (int j = 0; j < len; j++) {
                if (s1.charAt(i) == s2.charAt(j))
                    dp[i][j][1] = 1;
            }
        }
        /** start at i, j with len, split at k, 
         * if dp[i][j][k - i] && dp[k][j + k - i][len - (k - i)] is true
         * Or dp[i][j + len - (k - i)][k - i] && dp[k][j][len - (k - i)] is true
         * then dp[i][j][len] is true
         */
         for (int l = 2; l <= len; l++) {
             for (int i = 0; i <= len - l; i++) {
                 for (int j = 0; j <= len - l; j++) {
                     /** split at offset, i + offset, [i, offset), [i + offset, i + l) */
                     for (int offset = 1; offset < l; offset++) {
                         if (dp[i][j][offset] == 1 && dp[i + offset][j + offset][l - offset] == 1)
                            dp[i][j][l] = 1;
                        if (dp[i][j + l - offset][offset] == 1 && dp[i + offset][j][l - offset] == 1)
                            dp[i][j][l] = 1;
                     }
                 }
             }
         }
         return dp[0][0][len] == 1;
    }
    
    private boolean isSame(String s1, String s2) {
        char[] c1 = s1.toCharArray();
        char[] c2 = s2.toCharArray();
        Arrays.sort(c1);
        Arrays.sort(c2);
        s1 = new String(c1);
        s2 = new String(c2);
        return s1.equals(s2);
    }
}

这个解法的思路其实和解法1很类似,但是采用了dp
这也是我第一次意识到,dp到底是什么东西。
Dp就是一台机器,状态机。你给定,
初始状态,
一套状态跳转到下一个状态的规则。
然后让他运行。
固定次数后,出来的结果,或者说,模拟的结果,
就是你要的结果。
所以,dp的关键是,


  1. dp[][] 的物理意义,也有可能是三维数组
  2. 初始状态
  3. 状态间的转换规则

dp 的好处是,一旦你找到了这三个,那么,再难的题目,也可以在Np-complete的情况下完成。
但是,问题是,首先,很难抽象出dp的物理意义。
其次,无论情况多么简单,依然会用最复杂的dp去跑。也就是说,他的运行次数是固定的,不会中途结束或者跳一大段。

  1. Longest Palindromic Substring
    http://www.jianshu.com/p/4befb2f27ef1
    类似于这个,也是dp

greedy的话,就会根据具体情况具体分析,情况好的时候,会跑的很快,不会是一台规定不变的机器。

参考网页:
http://blog.csdn.net/ljiabin/article/details/44537523
http://www.jiuzhang.com/solutions/scramble-string/

Anyway, Good luck, Richardo!

My code:

public class Solution {
    /**
     * @param s1 A string
     * @param s2 Another string
     * @return whether s2 is a scrambled string of s1
     */
     
     private boolean checkScramble(String s1,int start1, String s2, int start2, int k, int [][][]visit) {
        if(visit[start1][start2][k] == 1)
            return true;
        if(visit[start1][start2][k] ==-1)
            return false;
        
        
        if (s1.length() != s2.length()) {
            visit[start1][start2][k] = -1;
            return false;
        }
        
        if (s1.length() == 0 || s1.equals(s2)) {
            visit[start1][start2][k] = 1;
            return true;
        }
        
        if (!isValid(s1, s2)) {
            visit[start1][start2][k] = -1;
            return false;
        }// Base Cases
        
        
        for (int i = 1; i < s1.length(); i++) {
            String s11 = s1.substring(0, i);
            String s12 = s1.substring(i, s1.length());
            
            String s21 = s2.substring(0, i);
            String s22 = s2.substring(i, s2.length());
            
            String s23 = s2.substring(0, s2.length() - i);
            String s24 = s2.substring(s2.length() - i, s2.length());
            
            if (checkScramble(s11,start1, s21, start2, i, visit) && checkScramble(s12, start1+i, s22, start2+i,k-i, visit))  {
                visit[start1][start2][k] = 1;
                return true;
            }
            
            if (checkScramble(s11,start1, s24, start2+k-i, i, visit) && checkScramble(s12,start1+i, s23,start2, k-i, visit))
            {
                visit[start1][start2][k] = 1;
                return true;
            }
        }
        visit[start1][start2][k] = -1;
        return false;
    }
    public boolean isScramble(String s1, String s2) {
        int len = s1.length();
        int [][][] visit = new int[len][len][len + 1];
        return checkScramble(s1,0,s2,0, len, visit);
    }
    
    
    private boolean isValid(String s1, String s2) {
        char[] arr1 = s1.toCharArray();
        char[] arr2 = s2.toCharArray();
        Arrays.sort(arr1);
        Arrays.sort(arr2);
        if (!(new String(arr1)).equals(new String(arr2))) {
            return false;
        }
        return true;
    }
}

reference:
http://www.jiuzhang.com/solutions/scramble-string/

算是 recursive DP + cache, 复杂度还是在 O(2 ^ n) 左右。
这道题目,加不加cache,最终的效果也差不多。

参考链接里,所谓的记忆化搜索,也不过是加了一层cache,
并没有彻底地改为: iteration DP

但是上面的第二种解法,正是iteration DP,三维的DP

说到底,这道题目还是算是 divide and conquer的一种变形。

Anyway, Good luck, Richardo! -- 08/24/2016

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,539评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,911评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,337评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,723评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,795评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,762评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,742评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,508评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,954评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,247评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,404评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,104评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,736评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,352评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,557评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,371评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,292评论 2 352

推荐阅读更多精彩内容

  • 背景 一年多以前我在知乎上答了有关LeetCode的问题, 分享了一些自己做题目的经验。 张土汪:刷leetcod...
    土汪阅读 12,743评论 0 33
  • 动态规划(Dynamic Programming) 本文包括: 动态规划定义 状态转移方程 动态规划算法步骤 最长...
    廖少少阅读 3,275评论 0 18
  • 回溯算法 回溯法:也称为试探法,它并不考虑问题规模的大小,而是从问题的最明显的最小规模开始逐步求解出可能的答案,并...
    fredal阅读 13,644评论 0 89
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,620评论 18 399
  • 分治方法 将问题划分成互不相交的子问题 递归地求解子问题 将子问题的解组合起来 动态规划(两个要素:最优子结构、子...
    superlj666阅读 498评论 0 0