【1】【leetcode-115 动态规划】 不一样的子序列

给定一个字符串 和一个字符串 T,计算在 S 的子序列中 T 出现的个数。数组

一个字符串的一个子序列是指,经过删除一些(也能够不删除)字符且不干扰剩余字符相对位置所组成的新字符串。(例如,"ACE" 是 "ABCDE" 的一个子序列,而 "AEC" 不是)spa

示例 1:3d

输入: S = , T = 解释:

以下图所示, 有 3 种能够从 S 中获得 。
(上箭头符号 ^ 表示选取的字母)


^^^^ ^^

^^ ^^^^

^^^ ^^^
"rabbbit""rabbit"
输出: 3"rabbit" 的方案rabbbitrabbbitrabbbit

示例 2:code

输入: S = , T = 解释:

以下图所示, 有 5 种能够从 S 中获得 。 
(上箭头符号 ^ 表示选取的字母)


^^ ^

^^    ^

^    ^^

  ^  ^^

    ^^^"babgbag""bag"
输出: 5"bag" 的方案babgbagbabgbagbabgbagbabgbagbabgbag

 

连接:https://www.nowcoder.com/questionTerminal/ed2923e49d3d495f8321aa46ade9f873
来源:牛客网

blog

咱们须要一个二维数组dp(i)(j)来记录长度为i的字串在长度为j的母串中出现的次数,这里长度都是从头算起的,并且遍历时,保持子串长度相同,先递增母串长度,母串最长时再增长一点子串长度重头开始计算母串。 字符串

首先咱们先要初始化矩阵,当子串长度为0时,全部次数都是1,当母串长度为0时,全部次数都是0.当母串子串都是0长度时,次数是1(由于都是空,相等)。接着,若是子串的最后一个字母和母串的最后一个字母不一样,说明新加的母串字母没有产生新的可能性,能够沿用该子串在较短母串的出现次数,因此dp(i)(j) = dp(i)(j-1)。若是子串的最后一个字母和母串的最后一个字母相同,说明新加的母串字母带来了新的可能性,咱们不只算上dp(i)(j-1),也要算上新的可能性。那么如何计算新的可能性呢,其实就是在既没有最后这个母串字母也没有最后这个子串字母时,子串出现的次数,咱们至关于为全部这些可能性都添加一个新的可能。因此,这时dp(i)(j) = dp(i)(j-1) + dp(i-1)(j-1)。下图是以rabbbit和rabbit为例的矩阵示意图。计算元素值时,当末尾字母同样,其实是左方数字加左上方数字,当不同时,就是左方的数字。 get

示意图


 

 

动归思想。加上一个新字母后,匹配的组数应该等于s、t都去掉该新字母成功匹配数 + s不加新字母时已经成功的组数it

个人:io

    public int numDistinct(String s, String t) {
        if (s == null || t == null || s.length()==0 || t.length() ==0) {
            return 0;
        }

        int[][] dp = new int[t.length()+1][s.length()+1];
        //初始化矩阵,当子串长度为0时,全部次数都是1,当母串长度为0时,全部次数都是0.空串""也是子串
        for (int i=0;i<t.length();i++) {
            dp[i][0] = 0;
        }
        for (int i=0;i<s.length();i++) {
            dp[0][i] = 1;
        }
        for (int i=1;i<=t.length();i++) {
            for (int j=1;j<=s.length();j++) {
                if (s.charAt(j-1) != t.charAt(i-1))
                    dp[i][j] = dp[i][j-1];
                else
                    dp[i][j] = dp[i-1][j-1] + dp[i][j-1];
            }
        }
        return dp[t.length()][s.length()];
    }
相关文章
相关标签/搜索