动规五部曲分析如下:
1、确定dp数组(dp table)以及下标的含义
dp[i][j]:以i-1为结尾的s子序列中出现以j-1为结尾的t的个数为dp[i][j]。
2、确定递推公式
这一类问题,基本是要分析两种情况
s[i - 1] 与 t[j - 1]相等
s[i - 1] 与 t[j - 1] 不相等
(1)当s[i - 1] 与 t[j - 1]相等时,dp[i][j]可以有两部分组成。
一部分是用s[i - 1]来匹配,那么个数为dp[i - 1][j - 1]。
一部分是不用s[i - 1]来匹配,个数为dp[i - 1][j]。
这里可能有同学不明白了,为什么还要考虑 不用s[i - 1]来匹配,都相同了指定要匹配啊。
例如: s:bagg 和 t:bag ,s[3] 和 t[2]是相同的,但是s也可以不用s[3]来匹配,即s[0]s[1]s[2]组成的bag。
当然也可以用s[3]来匹配,即:s[0]s[1]s[3]组成的bag。
所以当s[i - 1] 与 t[j - 1]相等时,dp[i][j] = dp[i - 1][j - 1] + dp[i - 1][j];
即如果s后还有可与t相匹配的及相当于有两种,如果没有就是一种
当s[i - 1] 与 t[j - 1]不相等时,dp[i][j]只有一部分组成,不用s[i - 1]来匹配,即:dp[i - 1][j]
所以递推公式为:dp[i][j] = dp[i - 1][j];
太详细了!!!!!!!!!!!!!!!!!!
举例推导dp数组
以s:“baegg”,t:"bag"为例,推导dp数组状态如下:
3、dp数组如何初始化
从递推公式dp[i][j] = dp[i - 1][j - 1] + dp[i - 1][j]; 和 dp[i][j] = dp[i - 1][j]; 中可以看出dp[i][0] 和dp[0][j]是一定要初始化的。
每次当初始化的时候,都要回顾一下dp[i][j]的定义,不要凭感觉初始化。
dp[i][0]表示什么呢?
dp[i][0] 以i-1为结尾的s可以随便删除元素,出现空字符串的个数。
那么dp[i][0]一定都是1,因为也就是把以i-1为结尾的s,删除所有元素,出现空字符串的个数就是1。
再来看dp[0][j],dp[0][j]:空字符串s可以随便删除元素,出现以j-1为结尾的字符串t的个数。
那么dp[0][j]一定都是0,s如论如何也变成不了t。
最后就要看一个特殊位置了,即:dp[0][0] 应该是多少。
dp[0][0]应该是1,空字符串s,可以删除0个元素,变成空字符串t。
初始化分析完毕,代码如下:
vector<vector<long long>> dp(s.size() + 1, vector<long long>(t.size() + 1));
for (int i = 0; i <= s.size(); i++) dp[i][0] = 1;
for (int j = 1; j <= t.size(); j++) dp[0][j] = 0; // 其实这行代码可以和dp数组初始化的时候放在一起,但我为了凸显初始化的逻辑,所以还是加上了。
4、确定遍历顺序
从递推公式dp[i][j] = dp[i - 1][j - 1] + dp[i - 1][j]; 和 dp[i][j] = dp[i - 1][j]; 中可以看出dp[i][j]都是根据左上方和正上方推出来的。
所以遍历的时候一定是从上到下,从左到右,这样保证dp[i][j]可以根据之前计算出来的数值进行计算。
代码如下:
for (int i = 1; i <= s.size(); i++) {
for (int j = 1; j <= t.size(); j++) {
if (s[i - 1] == t[j - 1]) {
dp[i][j] = dp[i - 1][j - 1] + dp[i - 1][j];
} else {
dp[i][j] = dp[i - 1][j];
}
}
}
完整代码如下:
public int numDistinct(String s, String t) {
int lens = s.length();
int lent = t.length();
//定义dp数组
int[][] dp = new int[lens+1][lent+1];
//初始化
for(int i=0;i<lens+1;i++) dp[i][0] = 1;
//遍历dp数组
for(int i=1;i<=lens;i++){
for(int j=1;j<=lent;j++){
if((s.charAt(i-1)) == (t.charAt(j-1)))
dp[i][j] = dp[i-1][j-1] + dp[i-1][j];
else
dp[i][j] = dp[i-1][j];
}
}
return dp[lens][lent];
}
public class li115 {
public static void main(String[] args) {
Scanner sc=new Scanner(System.in);
//String[] a=new String[10];
//String[] b=new String[10];
String a=sc.next();
String b=sc.next();
int s=a.length();
int t=b.length();
char aa[] = a.toCharArray();
char bb[] = b.toCharArray();
int[][] dp=new int[s+1][t+1];
//初始化
for (int i = 0; i < s+1; i++) {
dp[i][0]=1;
}
for (int j = 1; j <t+1; j++) {
dp[0][j]=0;
}
//确定遍历顺序
for (int i = 1; i <= s; i++) {
for (int j = 1; j <= t; j++) {
if (aa[i-1]==bb[j-1]) {
dp[i][j]=dp[i-1][j-1]+dp[i-1][j];
}else {
dp[i][j]=dp[i-1][j];
}
}
}
System.out.println(dp[s][t]);
}
}
作者:carlsun-2
链接:https://leetcode-cn.com/problems/distinct-subsequences/solution/dai-ma-sui-xiang-lu-115-bu-tong-de-zi-xu-q6uq/
来源:力扣(LeetCode)