给定一个字符串 S 和一个字符串 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()]; }