面试官:说说你对分而治之、动态规划的理解?区别?

开发 开发工具
分而治之是算法设计中的一种方法,就是把一个复杂的问题分成两个或更多的相同或相似的子问题,直到最后子问题可以简单的直接求解,原问题的解即子问题的解的合并.

[[429033]]

一、分而治之

分而治之是算法设计中的一种方法,就是把一个复杂的问题分成两个或更多的相同或相似的子问题,直到最后子问题可以简单的直接求解,原问题的解即子问题的解的合并

关于分而治之的实现,都会经历三个步骤:

  • 分解:将原问题分解为若干个规模较小,相对独立,与原问题形式相同的子问题
  • 解决:若子问题规模较小且易于解决时,则直接解。否则,递归地解决各子问题
  • 合并:将各子问题的解合并为原问题的解

实际上,关于分而治之的思想,我们在前面已经使用,例如归并排序的实现,同样经历了实现分而治之的三个步骤:

  • 分解:把数组从中间一分为二
  • 解决:递归地对两个子数组进行归并排序
  • 合并:将两个字数组合并称有序数组

同样关于快速排序的实现,亦如此:

分:选基准,按基准把数组分成两个字数组

解:递归地对两个字数组进行快速排序

合:对两个字数组进行合并

同样二分搜索也能使用分而治之的思想去实现,代码如下:

  1. function binarySearch(arr,l,r,target){ 
  2.     if(l> r){ 
  3.         return -1; 
  4.     } 
  5.     let mid = l + Math.floor((r-l)/2) 
  6.     if(arr[mid] === target){ 
  7.         return mid; 
  8.     }else if(arr[mid] < target ){ 
  9.         return binarySearch(arr,mid + 1,r,target) 
  10.     }else
  11.         return binarySearch(arr,l,mid - 1,target) 
  12.     } 

二、动态规划

动态规划,同样是算法设计中的一种方法,是一种在数学、管理科学、计算机科学、经济学和生物信息学中使用的,通过把原问题分解为相对简单的子问题的方式求解复杂问题的方法

常常适用于有重叠子问题和最优子结构性质的问题

简单来说,动态规划其实就是,给定一个问题,我们把它拆成一个个子问题,直到子问题可以直接解决

然后呢,把子问题答案保存起来,以减少重复计算。再根据子问题答案反推,得出原问题解的一种方法。

一般这些子问题很相似,可以通过函数关系式递推出来,例如斐波那契数列,我们可以得到公式:当 n 大于 2的时候,F(n) = F(n-1) + F(n-2) ,

f(10)= f(9)+f(8),f(9) = f(8) + f(7)...是重叠子问题,当n = 1、2的时候,对应的值为2,这时候就通过可以使用一个数组记录每一步计算的结果,以此类推,减少不必要的重复计算

适用场景

如果一个问题,可以把所有可能的答案穷举出来,并且穷举出来后,发现存在重叠子问题,就可以考虑使用动态规划

比如一些求最值的场景,如最长递增子序列、最小编辑距离、背包问题、凑零钱问题等等,都是动态规划的经典应用场景

关于动态规划题目解决的步骤,一般如下:

  • 描述最优解的结构
  • 递归定义最优解的值
  • 按自底向上的方式计算最优解的值
  • 由计算出的结果构造一个最优解

三、区别

动态规划算法与分治法类似,其基本思想也是将待求解问题分解成若干个子问题,先求解子问题,然后从这些子问题的解得到原问题的解

与分治法不同的是,适合于用动态规划求解的问题,经分解得到子问题往往「不是互相独立」的,而分而治之的子问题是相互独立的

若用分治法来解这类问题,则分解得到的子问题数目太多,有些子问题被重复计算了很多次

如果我们能够保存已解决的子问题的答案,而在需要时再找出已求得的答案,这样就可以避免大量的重复计算,节省时间

综上,可得:

动态规划:有最优子结构和重叠子问题

分而治之:各子问题独立

参考文献

https://baike.baidu.com/item/%E5%8A%A8%E6%80%81%E8%A7%84%E5%88%92/529408

 

https://juejin.cn/post/6951922898638471181

 

责任编辑:武晓燕 来源: JS每日一题
相关推荐

2021-08-09 07:47:40

Git面试版本

2021-11-25 10:18:42

RESTfulJava互联网

2024-07-25 18:20:03

2020-12-01 08:47:36

Java异常开发

2020-06-12 15:50:56

options前端服务器

2021-08-19 08:36:22

Git ResetGit Revert版本

2021-08-17 07:15:16

Git RebaseGit Merge面试

2020-08-17 07:40:19

消息队列

2021-11-08 11:32:01

观察

2021-09-16 07:52:18

算法应用场景

2021-09-07 08:33:27

JavaScript TypeScript 函数

2019-05-10 10:50:04

Spring AOPJDK动态代理CGLIB动态代理

2021-11-02 22:04:58

模式

2022-02-21 17:24:18

序列化对象存储

2021-08-16 08:33:26

git

2021-11-04 06:58:32

策略模式面试

2021-11-10 07:47:49

组合模式场景

2021-11-03 14:10:28

工厂模式场景

2020-12-04 06:27:04

序列化面试官Java

2021-11-05 07:47:56

代理模式对象
点赞
收藏

51CTO技术栈公众号