动态规划的一般思路是分为四步,即:寻找最优子结构、递归定义最优子结构、自底向上求解最优子结构和构造最优解。

接下来我列举出几个常见的动态规划面试题进行说明。

(1)数学三角形:比较简单,直接贴一个我看到的讲得最清楚的文章,http://blog.csdn.net/baidu_28312631/article/details/47418773

(2)LIS:最长上升子序列问题。

  思路1:

  其实就是寻找f(n)和f(n-1)之间的关系,对于一个序列,f(n)要么等于f(n-1),要么等于f(n-1)+1, 但是什么情况下相等,这是需要思考清楚的,也是解决这个问题的关键。

  通过自己写一写test case,可以发现这种递推式无法很方便地写出来,因为第n个数和第n-1个数的关系无法推导f(n), 但是我们换个思路转换原问题的话,马上可以得到这样一个递推式:

              

  这里对问题进行了转换,变成了:序列加入一个数后,包含这个数的最长上升子序列。举个例子,如果加入的数小于序列其他值,那么显然有L(k) = 1。 这是一个递推式,我们很容易能自底向上求得L(k).

  这个问题和原问题并不等价,但是我们可以发现既然是最长递增子序列,那么只要找到这个问题中L(k)的最大值即可。原问题的f(k) = max(L(k)) , 这个才是原问题的解。

  思路2:(奇妙的解法,复杂度O(n*logn), 看了几篇文章发现要么不解释,要么解释很复杂,我这里简洁描述一下,应该挺好懂)

  由于一个序列的上升子序列有很多,那么我们如何判断新加入一个数后,最长子序列是否增长了呢?我们可以记录相同长度的子序列中所有最大值的最小值,只要加入一个数,就遍历不同长度子序列的最小值,判断新加入的数是否大于这个最小值就能得到最长子序列。

  所以,创建一个长度为n的数组来存这些不同长度的最小值,每加入一个数就进行二分搜索并更新这个数组,代码如下:

  

int LIS(int* a, int n)
{
int dp[n];
const int inf = 0x7FFFFFFF;
fill(dp, dp + n, inf);
for(int i = ; i < n; i++) {
*upper_bound(dp,dp + n, a[i]) = a[i];//二分搜索,返回一个非递减序列[first, last)中的第一个大于值val的位置
}
return (lower_bound(dp, dp + n, inf)-dp);//二分搜索,返回一个非递减序列[first, last)中的第一个大于等于值val的位置
}

(3)LCS:最长公共子序列问题,比如求两个字符串的最长公共字符串长度。

比如字符串1:BDCABA;字符串2:ABCBDAB

则这两个字符串的最长公共子序列长度为4,最长公共子序列是:BCBA

思路:可以推导出如下递归公式:

//递归解法
int LCS(const char* s1, const char* s2, int i, int j)
{
if(i< || j<)
return ;
if(s1[i] == s2[j])
{
return LCS(s1, s2, i-, j-)+;
}
else
{
return max(LCS(s1, s2, i, j-),LCS(s1, s2, i-, j));
}
}
//循环解法
int LCS2(string s1, string s2)
{
int len1 = s1.length();
int len2 = s2.length();
int dp[len1+][len2+];
for(int i=;i<=len1;++i)
fill(dp[i], dp[i]+len2+, );
for(int i=;i<=len1;++i)
{
for(int j=;j<=len2;++j)
{
if(s1[i-]==s2[j-])
dp[i][j] = dp[i-][j-]+;
else
dp[i][j] = max(dp[i-][j],dp[i][j-]);
}
}
return dp[len1][len2];
}

(4)CSD(编辑距离):

字符串的编辑距离,又称为Levenshtein距离,由俄罗斯的数学家Vladimir Levenshtein在1965年提出。是指利用字符操作,把字符串A转换成字符串B所需要的最少操作数。其中,字符操作包括:

  • 删除一个字符
  • 插入一个字符
  • 修改一个字符

例如对于字符串"if"和"iff",可以通过插入一个'f'或者删除一个'f'来达到目的。

思路:递推公式如下

i>0,j>0的公式分别是删除一个字符,插入一个字符,修改一个字符后的操作次数。其它的边界条件就容易理解了。递归代码如下,循环版本的也很好写。

int min3(int a, int b, int c)
{
return min(min(a,b),c);
} int CSD(const char* s1, const char* s2, int i, int j)
{
if(i< && j<)
return ;
else if(i<)
return j+;
else if(j<)
return i+;
return min3(CSD(s1,s2,i-,j)+, CSD(s1,s2,i,j-)+, CSD(s1,s2,i-,j-)+(s1[i]==s2[j]?:));
}

动态规划面试题基础合集1--数学三角形,LIS , LCS, CSD的更多相关文章

  1. es6常用基础合集

    es6常用基础合集 在实际开发中,ES6已经非常普及了.掌握ES6的知识变成了一种必须.尽管我们在使用时仍然需要经过babel编译. ES6彻底改变了前端的编码风格,可以说对于前端的影响非常巨大.值得 ...

  2. Java面试题资料合集

    Java面试题资料合集 2021年最新版--Java+最常见的+200++面试题汇总+答案总结汇总.pdf BIO,NIO,AIO,Netty面试题 35道.pdf Dubbo面试题 47道.pdf ...

  3. 前端基础进阶(十四):es6常用基础合集

    在实际开发中,ES6已经非常普及了.掌握ES6的知识变成了一种必须.尽管我们在使用时仍然需要经过babel编译. ES6彻底改变了前端的编码风格,可以说对于前端的影响非常巨大.值得高兴的是,如果你熟悉 ...

  4. MySQL基础合集

     我的小站 1.MySQL的优势 运行速度快 使用成本低 可移植性强 适用用户广 2.MySQL的运行机制 一个SQL语句,如select * from tablename ,从支持接口进来后,进入连 ...

  5. Linux 命令基础合集

    下面总结的是在使用 Linux 最常用的命令: 常用指令 ls        显示文件或目录 -l           列出文件详细信息l(list) -a          列出当前目录下所有文件及 ...

  6. 备战金三银四!一线互联网公司java岗面试题整理:Java基础+多线程+集合+JVM合集!

    前言 回首来看2020年,真的是印象中过的最快的一年了,真的是时间过的飞快,还没反应过来年就夸完了,相信大家也已经开始上班了!俗话说新年新气象,马上就要到了一年之中最重要的金三银四,之前一直有粉丝要求 ...

  7. [题解+总结]NOIP动态规划大合集

    1.前言 NOIP2003-2014动态规划题目大合集,有简单的也有难的(对于我这种动态规划盲当然存在难的),今天就把这些东西归纳一下,做一个比较全面的总结,方便对动态规划有一个更深的理解. 2.NO ...

  8. NOIP动态规划大合集

    1.前言 NOIP2003-2014动态规划题目大合集,有简单的也有难的(对于我这种动态规划盲当然存在难的),今天就把这些东西归纳一下,做一个比较全面的总结,方便对动态规划有一个更深的理解. 2.NO ...

  9. Struts+Hibernate+Spring面试题合集及答案

    Struts+Hibernate+Spring面试题合集及答案 Struts+Hibernate+Spring面试题合集 1 1. Hibernate部分 2 1.1. Hibernate工作原理 2 ...

随机推荐

  1. tar 命令显示进度条

    实现该功能需要安装 pv,然后把需要处理的数据通过管道传给 pv,最后再进行操作. 传给 pv 的目的是为了知道已经处理的数据量大小,同时需要通过 -s 指定总共需要处理的数据量大小. pv 的安装一 ...

  2. GCC、GNU C、C99、ANSI C

    ANSI C ANSI C是由美国国家标准协会(ANSI)及国际标准化组织(ISO)推出的关于C语言的标准.ANSI C 标准同时规定了 C 标准库. ANSI C发展历史 C 的第一个标准是由ANS ...

  3. Java入门:读写文本文件

    文本文件的读写是学习java必须掌握的一项基本技术,因为在项目中时常会涉及到文本文件的读写. 一.使用FileWriter写文件 1.FileWriter类 [功能] FileWriter类专门用来写 ...

  4. Docker管理工具-Swarm

    一.Swarm介绍 Swarm是Docker公司在2014年12月初发布的一套较为简单的工具,用来管理Docker集群,它将一群Docker宿主机变成一个单一的,虚拟的主机.Swarm使用标准的Doc ...

  5. 多页面应用 VS 单页面应用

    多页面应用 每一次页面跳转,后端都会返回一个新的HTML文件, 优点:首屏时间快(只经历了一个HTTP请求),SEO效果好 缺点:页面切换慢 单页面应用 进行页面之间跳转时,并不去加载HTML文件,而 ...

  6. bzoj千题计划161:bzoj1589: [Usaco2008 Dec]Trick or Treat on the Farm 采集糖果

    http://www.lydsy.com/JudgeOnline/problem.php?id=1589 tarjan缩环后拓扑排序上DP #include<cstdio> #includ ...

  7. 并查集:POJ No1703 Find them, Catch them

    题目链接:http://poj.org/problem?id=1703 题意:两个坏蛋属于不同的组织,给出两个坏蛋判定是否一个组织. 题解:已知每次输入的两个帮派人员 x, y; 合并 (x, y + ...

  8. e.getMessage() e.printStackTrace() 和e.printStackTrace() 小结

    1 e.getMessage() ;    只会获得异常的名称.比如说NullPoint 空指针,就告诉你说是空指针 2.e.toString():  获得异常种类和错误信息 3.e.printSta ...

  9. yum安装_yum命令的相关操作

    2017年1月11日, 星期三 yum安装的四种方式 一.默认:从国外下载 二.国内:从阿里获取  http://mirrors.aliyun.com 1. cd /etc/yum.repos.d 2 ...

  10. Javascript Ajax异步读取RSS文档

    RSS 是一种基于 XML的文件标准,通过符合 RSS 规范的 XML文件可以简单实现网站之间的内容共享.Ajax 是Asynchronous JavaScript and XML的缩写.通过 Aja ...