博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Distinct Subsequences
阅读量:6123 次
发布时间:2019-06-21

本文共 1236 字,大约阅读时间需要 4 分钟。

  对于有两个字符串的问题,比较常见的方法就是递归或者动态规划,而通常递归会超时。这题一看就知道递归肯定会超时,DP就成为解题利器了。前面总结过,DP一个很重要的问题是要解决意义问题:即dp[i][j]到底表示什么。对这题而言,容易想到dp[i][j]就表示模板串T的前i个字符在母串S的前j个字符中出现的次数。通常,解决了意义问题,接下来的问题就简单了,即写出递归方程。但是这题仍然让我费了好大劲,因为这题的递归方程首先不好找,其次找到了也难以理解。

  一. 找递归方程

  如果思路不是很清楚,那先把图画一画。

  仔细找能够发现:

  方程a. 如果T[i-1]!=S[j-1],则DP[i][j]=DP[i][j-1]。

  方程b. 如果T[i-1]==S[j-1],则DP[i][j]=DP[i][j-1]+DP[i-1][j-1]。

 

  二. 理解思路

  关于上述递归方程,方程a比较容易理解,而方程b就比较难。对于方程b的理解,以下面为例子:

  假设现在现在匹配Ta->Sa,模式串的末尾和母串的末尾相等,都是字符a。则它的值应该是T->S加上Ta->S。为什么是这样?T->S表示两个末尾的a严格对应,Ta->S则表示模式串的a对应的是S中已经包含的a。这样两部分加起来就是所有的出现的子串。

  基于上述说明,不难写出代码:

public class Solution {    public int numDistinct(String S, String T) {        int[][]dp=new int[T.length()+1][S.length()+1];        int i,j;        for(j=0;j<=S.length();j++)        {            dp[0][j]=1;        }        for(i=1;i<=T.length();i++)        {            dp[i][0]=0;        }        for(i=1;i<=T.length();i++)        {            for(j=1;j<=S.length();j++)            {                dp[i][j]=dp[i][j-1];                if(T.charAt(i-1)==S.charAt(j-1)){                    dp[i][j]+=dp[i-1][j-1];                }            }        }        return dp[T.length()][S.length()];    }}

  

转载于:https://www.cnblogs.com/zhizhizhiyuan/p/3584407.html

你可能感兴趣的文章
数据结构中常见的树(BST二叉搜索树、AVL平衡二叉树、RBT红黑树、B-树、B+树、B*树)...
查看>>
PHP读取日志里数据方法理解
查看>>
第五十七篇、AVAssetReader和AVAssetWrite 对视频进行编码
查看>>
Vivado增量式编译
查看>>
一个很好的幻灯片效果的jquery插件--kinMaxShow
查看>>
微信支付签名配置正确,但返回-1,调不出支付界面(有的手机能调起,有的不能)...
查看>>
第二周例行报告
查看>>
vue实现点击展开,点击收起
查看>>
如何使frame能居中显示
查看>>
第k小数
查看>>
构建之法阅读笔记三
查看>>
写给对前途迷茫的朋友:五句话定会改变你的人生
查看>>
并行程序设计学习心得1——并行计算机存储
查看>>
JAVA入门到精通-第86讲-半双工/全双工
查看>>
bulk
查看>>
js document.activeElement 获得焦点的元素
查看>>
C++ 迭代器运算
查看>>
【支持iOS11】UITableView左滑删除自定义 - 实现多选项并使用自定义图片
查看>>
day6-if,while,for的快速掌握
查看>>
JavaWeb学习笔记(十四)--JSP语法
查看>>