115. Distinct Subsequences(不同的子序列)

115. Distinct Subsequences

OK,乍一看题目是我会写的题目,没错字符串匹配问题,仔细一想对不起,老样子状态转移过程写的有点问题,仔细想一想加上讨论区的醍醐灌顶,明白。字符串匹配问题,我们同样从后往前考虑,假设当前字符串 s [0, i),也就是从0到i-1的字符和字符串 t [0, j)已经匹配完成,并且dp[ i ][ j ]存的是当前有多少符合要求的子序列。那么考虑,当s[ i ]和t[ j ]字符相同和不同时这两种情况时候dp [ i + 1 ][ j+1 ]的变化。

注意上面由于我个人的习惯dp[ i ][ j ]代表的字符串 s [0,i-1]和 t [0, j-1]这个在我以前的博客中也提起过。好了我么来讨论两种情况dp的变化。

1. 当s [ i-1 ] == t [ j-1 ]
由于当前匹配的两个字符相同,这相当于两个字符串各自往前倒退一次匹配的情况:也就是dp[ i-1 ][ j-1 ],也就是说dp[ i-1 ][ j-1 ]中有多少个符合情况的那么现在两个字符相同在此基础上再往前进一位完全相等,这是一种情况。

dp[ i ][ j ] = dp[ i-1 ][ j-1 ] + ?

为什么加上?是因为还缺少一项,如果仅仅等于上面是不是却少了一种情况,上面dp[ i-1 ][ j-1 ]是不包含 t [i-1]这一个字符,那么包含字符 t [i-1]有多少也需要加上。

dp[ i ][ j ] = dp[ i-1 ][ j-1 ] + dp[ i-1 ][ j ]

2.当s [ i-1 ] != t [ j-1 ]
那么当不相等时候,如果理解了上面的转移过程,那么这种就很好理解了,因为不相等,那么他只能等于dp[ i-1 ][ j ]。

dp[ i ][ j ] = dp[ i-1 ][ j ]

最后我们需要考虑一下边界问题,要考虑这么一个问题如果t字符串是一个空串,那么这个答案就是1,因为s子序列可以是空串所以有一个。也就是dp[i][0]=1。

那么我们上代码去具体看一下

class Solution {
public:
	int numDistinct(string s, string t) {
		int m = s.size();
		int n = t.size();
		vector<vector<long>> dp(m + 1, vector<long>(n + 1, 0));
		for (int i = 0; i <= m; i++)
			dp[i][0] = 1;
		for (int i = 1; i <= m; i++)
		{
			for (int j = 1; j <= n; j++)
			{
				if (s[i - 1] == t[j - 1])
					dp[i][j] = dp[i-1][j] + dp[i - 1][j - 1];
				else
					dp[i][j] = dp[i-1][j];
			}
		}
		return int(dp[m][n]);
	}
};
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值