算法-分治算法

妖狐艹你老母 2024-04-08 10:08 161阅读 0赞

一、分治

1、定义:分治,也就是分而治之。

它的一般步骤是:

① 将原问题分解成若干个规模较小的子问题(子问题和原问题的结构一样,只是规模不一样)

② 子问题又不断分解成规模更小的子问题,直到不能再分解(直到可以轻易计算出子问题的解)

③ 利用子问题的解推导出原问题的解

分治策略非常适合用递归

需要注意的是:子问题之间是相互独立的

2、分治的应用

  • 快速排序
  • 归并排序
  • Karatsuba算法(大数乘法)

3、分治时间复杂度的计算–主定理

800d948f368a29f98792b4e349627502.png

4、最大连续子序列和

子序列:按照原序列的排序顺序,从原序列取出部分元素

连续子序列:按照原序列的排序顺序,连续地从原序列取出部分元素

  • 举例:

    原序列:–2、1、–3、4、–1、2、1、–5、4

    子序列可以是:–2、1、1、4 还可以是:4、1、4 还可以是:2、1、–5、4 等等

    连续子序列可以是:–2、1、–3、4、–1 还可以是:–3、4、–1 还可以是:2、1、–5、4 等等

子串、子数组、子区间必须是连续的,子序列是可以不连续的

解法:分治

◼ 将序列均匀地分割成 2 个子序列

  • [begin , end) = [begin , mid) + [mid , end),mid = (begin + end) >> 1

◼ 假设 [begin , end) 的最大连续子序列和是 S[i , j) ,那么它有 3 种可能

  • [i , j) 存在于 [begin , mid) 中,同时 S[i , j) 也是 [begin , mid) 的最大连续子序列和
  • [i , j) 存在于 [mid , end) 中,同时 S[i , j) 也是 [mid , end) 的最大连续子序列和
  • [i , j) 一部分存在于 [begin , mid) 中,另一部分存在于 [mid , end) 中

    • [i , j) = [i , mid) + [mid , j)
    • S[i , mid) = max { S[k , mid) },begin ≤ k < mid
    • S[mid , j) = max { S[mid , k) },mid < k ≤ end

38acd225b5dd3a1511354b70f01d391a.png

对于解:只在左边或者只在右边,可以直接使用 递归

对于解:在中间,一部分在左边,一部分在右边的情况:

091a0ff6eab1f3bc9edfa5a65b242af5.png

  • 需要先 从中间mid开始统计[mid - 1, 左边某个元素] 统计出左边的最大值
  • 再从中间mid开始统计[mid, 右边某个元素] 统计出右边的最大值
  • 然后左边最大值+右边最大值,就是 横跨两个区域的解
  1. static int maxSubArray(int[] nums) {
  2. if(nums == null || nums.length == 0) return 0;
  3. return maxSubArray(nums, 0, nums.length);
  4. }
  5. /**
  6. * 分治法
  7. */
  8. private static int maxSubArray(int[] nums, int begin, int end) {
  9. if(end - begin < 2) return nums[begin];
  10. int mid = (begin + end) >> 1;
  11. // 要从中间mid开始统计[mid - 1, 左边某个元素]、[mid, 右边某个元素]的连续最大子序列和
  12. int leftMax = nums[mid - 1];
  13. int leftSum = leftMax;
  14. for(int i = mid - 2; i >= begin; i--) {
  15. leftSum += nums[i];
  16. leftMax = Math.max(leftMax, leftSum);
  17. }
  18. int rightMax = nums[mid];
  19. int rightSum = rightMax;
  20. for(int i = mid + 1; i < end; i++) {
  21. rightSum += nums[i];
  22. rightMax = Math.max(rightMax, rightSum);
  23. }
  24. int midMax = leftMax + rightMax;
  25. return Math.max(midMax, //中间的最大连续子序列和
  26. Math.max(maxSubArray(nums, begin, mid -1), maxSubArray(nums,mid, end))); //左边、右边的最大连续子序列和
  27. }

如果本文对你有帮助的话记得给一乐点个赞哦,感谢!

发表评论

表情:
评论列表 (有 0 条评论,161人围观)

还没有评论,来说两句吧...

相关阅读

    相关 算法-分治算法

    一、分治 1、定义:分治,也就是分而治之。 它的一般步骤是: ① 将原问题分解成若干个规模较小的子问题(子问题和原问题的结构一样,只是规模不一样)

    相关 算法思想-分治算法

    tip:作为程序员一定学习编程之道,一定要对代码的编写有追求,不能实现就完事了。我们应该让自己写的代码更加优雅,即使这会费时费力。 推荐:[体系化学习Java(Java面试专

    相关 分治算法

    问题描述 在一个2k×2k 个方格组成的棋盘中,恰有一个方格与其它方格不同,称该方格为一特殊方格,且称该棋盘为一特殊棋盘。在棋盘覆盖问题中,要用图示的L型骨牌覆盖给定的特

    相关 分治算法

    [概述参考博客地址][Link 1] [算法笔记总目录][Link 2] 一、分治算法的设计思想: 1. 分–将问题分解为规模更小的子问题; 2. 治–将这些规

    相关 分治算法

    划分问题:整个问题划分成多个无关联的子问题。 递归求解:递归调用求解各个子问题。 合并问题:合并子问题的解,形成原始问题的解。 -------------------

    相关 分治算法

    分治算法的基本思想是将一个规模为N的问题分解为K个规模较小的子问题,这些子问题相互独立且与原问题性质相同。求出子问题的解,就可得到原问题的解。即一种分目标完成程序算法,简单问题

    相关 分治算法

    分治算法的基本思想是将一个规模为N的问题分解为K个规模较小的子问题,这些子问题相互独立且与原问题性质相同。求出子问题的解,就可得到原问题的解。即一种分目标完成程序算法,简单问题

    相关 分治算法

    一 分治算法介绍 1 分治法是一种很重要的算法 字面上的解释是“分而治之,就是把一个复杂的问题分成两个或更多的相同或相似的子问题,再把子问题分成更小的子问题……,直