您当前的位置:首页 > 电脑百科 > 程序开发 > 算法

大厂高级程序员必备算法,看似简单的二分查找,您了解多少?

时间:2022-07-04 10:51:09  来源:  作者:程序猿星球

一、基础概念

1、Sorted(单调递增or单调递减)

2、Bounded(存在上下界)

3、Accessible by index(能够通过索引访问,数组适合,but链表不适合)

二分查找是一种在每次比较之后将查找空间一分为二的算法。每次需要查找集合中的索引或元素时,都应该考虑二分查找。如果集合是无序的,我们可以总是在应用二分查找之前先对其进行排序。

二分查找一般由三个主要部分组成:

1、预处理--如果有集合未排序,则进行排序。

2、二分查找--使用循环或递归在每次比较后将查找空间划分为两半

3、后处理--在剩余空间中确定可行的候选者

例如:给定一个 n 个元素有序的(升序)整型数组 nums 和一个目标值 target ,写一个函数搜索 nums 中的 target,如果目标值存在返回下标,否则返回 -1。

示例 1:

输入: nums = [-1,0,3,5,9,12], target = 9
输出: 4
解释: 9 出现在 nums 中并且下标为 4

示例 2:

输入: nums = [-1,0,3,5,9,12], target = 2
输出: -1
解释: 2 不存在 nums 中因此返回 -1

提示:

1、你可以假设 nums 中的所有元素是不重复的。

2、n 将在 [1, 10000]之间。

3、nums 的每个元素都将在 [-9999, 9999]之间。

 

step01: 设定初始值 left和 right:

大厂高级程序员必备算法,看似简单的二分查找,您了解多少?

 

step02: 设定中间指针 mid = left + (right-left)/2:

大厂高级程序员必备算法,看似简单的二分查找,您了解多少?

 

step03:

大厂高级程序员必备算法,看似简单的二分查找,您了解多少?

 

step04:

大厂高级程序员必备算法,看似简单的二分查找,您了解多少?

 

step05:

大厂高级程序员必备算法,看似简单的二分查找,您了解多少?

 

step06:

大厂高级程序员必备算法,看似简单的二分查找,您了解多少?

 

step07:

大厂高级程序员必备算法,看似简单的二分查找,您了解多少?

 

二、通用模版

2.1 模版一

var binarySeach(nums,target){
  if(nums == null|| nums.length ==0){
    return -1;
  }
  let left =0,right=nums.length- 1; //初始条件
  while(left<=right){ //终止条件 left>right
    let mid = Math.floor(left + (right-left)/2); 
    if(nums[mid] == target){ 
      return mid;
    } else if(nums[mid]<target){
      left = mid + 1; //向右查找
    }else{
      right = mid - 1; //向左查找
    }
  }
  return -1;
}

关键属性:

  • 二分查找的最基础和最基本的形式
  • 查找条件可以在不与元素的两侧进行比较的情况下确定(或使用它周围的特定元素)
  • 不需要后处理,因为每一步中,你都在检查是否找到了元素。如果到达末尾,则指导未找到该元素。

2.2 模版二

var binarySeach(nums,target){
  if(nums == null|| nums.length ==0){
    return -1;
  }
  
  let left =0,right=nums.length; //初始条件
  
  while(left<right){ //终止条件 left>=right
    let mid = Math.floor(left + (right-left)/2);  //阻止(left+right)溢出
    if(nums[mid] == target){ 
      return mid;
    } else if(nums[mid]<target){
      left = mid + 1; //向右查找
    }else{
      right = mid; //向左查找
    }
  }
  if(left !=nums.length && nums[left] == target) return left;
  return -1;
}

模版二:是二分查找的高级模板。它用于查找需要访问数组中当前索引及其直接右邻居索引的元素或条件。

关键属性:

  • 一种实现二分查找的高级方法
  • 查找条件需要访问元素的直接右邻居
  • 使用元素的右邻居来确定是否满足条件,并决定是向左还是向右
  • 保证查找空间在每一步中至少有2个元素
  • 需要进行后处理,当你剩下一个元素的时候,循环/递归结束。需要评估剩余元素是否符合条件

2.3 模版三

var binarySeach(nums,target){
  if(nums == null|| nums.length ==0){
    return -1;
  }
  
  let left =0,right=nums.length-1; //初始条件
  
  while(left+1 < right){ //终止条件 left+1 == right
    let mid = Math.floor(left + (right-left)/2);  //阻止(left+right)溢出
    if(nums[mid] == target){ 
      return mid;
    } else if(nums[mid]<target){
      left = mid; //向右查找
    }else{
      right = mid; //向左查找
    }
  }
  if(nums[left] == target) return left;
  if(nums[right] == target) return right;
  return -1;
}

模版3是二分法查找的另一种独特形式。 它用于搜索需要访问当前索引及其在数组中的直接左右邻居索引的元素或条件。

关键属性:

  • 实现二分法查找的另一种方法
  • 搜索条件需要访问元素的直接左右邻居
  • 使用元素的邻居来确定它是向右还是向左
  • 保证查找空间在每个步骤中至少有3个元素
  • 需要进行后处理。当剩下2个元素,循环/递归结束。

2.4、模版分析:

大厂高级程序员必备算法,看似简单的二分查找,您了解多少?

 

这三个模版的不同之处在于:

  • 左、中、右索引的分配
  • 循环或递归终止条件
  • 后处理的必要性

其中模版一和模版三是最常用的, 几乎所有二分查找问题都可以用其中之一亲送实现。 模版二更高级一些, 用于解决某些类型的问题。

这 3 个模板中的每一个都提供了一个特定的用例:

模板 1 (left <= right)

  • 二分查找的最基础和最基本的形式。
  • 查找条件可以在不与元素的两侧进行比较的情况下确定(或使用它周围的特定元素)。
  • 不需要后处理,因为每一步中,你都在检查是否找到了元素。如果到达末尾,则知道未找到该元素。

模板 2 (left < right)

  • 一种实现二分查找的高级方法。
  • 查找条件需要访问元素的直接右邻居。
  • 使用元素的右邻居来确定是否满足条件,并决定是向左还是向右。
  • 保证查找空间在每一步中至少有 2 个元素。
  • 需要进行后处理。 当你剩下 1 个元素时,循环 / 递归结束。 需要评估剩余元素是否符合条件。

模板 3 (left + 1 < right)

  • 实现二分查找的另一种方法。
  • 搜索条件需要访问元素的直接左右邻居。
  • 使用元素的邻居来确定它是向右还是向左。
  • 保证查找空间在每个步骤中至少有 3 个元素。
  • 需要进行后处理。 当剩下 2 个元素时,循环 / 递归结束。 需要评估其余元素是否符合条件。

三、复杂度计算

3.1、时间复杂度

O(log n) :

因为二分查找是通过对查找空间中间的值应用一个条件来操作的,并因此将查找空间折半,在更糟糕的情况下,我们将不得不进行O(log n) 次比较, 其中n是集合中元素的数目。

为什么是log n?

  • 二分查找是通过将现有数组一分为二来执行的
  • 因此,每次调用子例程(或完成一次迭代)时,其大小都会减少到现有部分的一半。
  • 首先N变成 N/2,然后又变成N/4,然后继续下去,直到找到元素或尺寸变为1。
  • 迭代的最大次数是log N(base2)。
第一次: n/2
第二次: n/2^2
第三次: n/2^3
...
第k次: n/2^k
大厂高级程序员必备算法,看似简单的二分查找,您了解多少?

 


大厂高级程序员必备算法,看似简单的二分查找,您了解多少?

 

3.2、空间复杂度

O(1)

虽然二分查找确实需要跟踪 3 个指标,但迭代解决方案通常不需要任何其他额外空间,并且可以直接应用于集合本身,因此需要 O(1) 或常量空间。

四、二分查找的应用

  • 二分调试法

把所有潜在的问题都用类似“数组二分查找”的方式把代码遍历一遍,不断缩小问题的范围,最终找到问题原因。

通过二分法,我们可以快速缩小问题范围,这样一来调试的效率也就上去了。

  • Kafka 采用二分法找数据

二分查找相关系列题:

大厂高级程序员必备算法,看似简单的二分查找,您了解多少?

 



Tags:算法   点击:()  评论:()
声明:本站部分内容及图片来自互联网,转载是出于传递更多信息之目的,内容观点仅代表作者本人,如有任何标注错误或版权侵犯请与我们联系(Email:2595517585@qq.com),我们将及时更正、删除,谢谢。
▌相关推荐
一、什么是递归?自己调用自己,当业务逻辑符合以下三个条件的时候,就可以考虑使用递归来实现。 一个问题可以分解为多个子问题; 当前问题与其子问题除了数据规模不同外,求解思路...【详细内容】
2022-10-07  Tags: 算法  点击:(10)  评论:(0)  加入收藏
相信大家都知道,在短视频中,抖音和tiktok流量最大.使用用户最多的平台之一就是所谓的国内抖音,国外抖音。tiktok,虽然这两个平台的实现方式几乎是一样的,但算法是人们比较纠结,不...【详细内容】
2022-10-06  Tags: 算法  点击:(6)  评论:(0)  加入收藏
最经典最常用的排序算法有:冒泡排序、插入排序、选择排序、归并排序、快速排序、计数排序、基数排序和桶排序。这些排序算法可以按照时间复杂度分为三类: O(n^2)&mdash;&mdash...【详细内容】
2022-10-01  Tags: 算法  点击:(12)  评论:(0)  加入收藏
导读:退无可退那就无需再退,蹬鼻子上脸不能惯,解放军的态度已经说明一切,主权这种事没有任何讨价还价的余地。编辑:武穆 桃子文章来源:新智元(AI_era)众所周知,2020年的大选,是拜登...【详细内容】
2022-09-28  Tags: 算法  点击:(21)  评论:(0)  加入收藏
9月6日,2022中国(郑州)国际期货论坛在线上开幕。郑商所理事长熊军在当日上午的主论坛上致辞时表示,今年是中国期货市场发展历史上具有里程碑意义的一年。期货和衍生品法的顺利出...【详细内容】
2022-09-07  Tags: 算法  点击:(35)  评论:(0)  加入收藏
1、A GPU accelerated Genetic Algorithm for the Construction of Hadamard Matrices Andras Balogh, Raven Ruiz这篇论文使用遗传算法来构建Hadamard矩阵。 生成随机矩...【详细内容】
2022-09-06  Tags: 算法  点击:(59)  评论:(0)  加入收藏
“外卖骑手的工作现状并没有得到实质改变”文 | Tom我从7月11日开始加入美团外卖。到8月10日,共计工作了31天。回看这段时间的经历,除了手上留下的黝黑的晒痕以外,有第一次送单...【详细内容】
2022-08-28  Tags: 算法  点击:(64)  评论:(0)  加入收藏
你在刷抖音的时候,有没有发现,抖音从来不会给你推送相同内容的视频?你可能会想,这有啥难的,给每个人都存一个记录,以后推送的时候避开就好了呀。nononono!可没有这么简单啊!海量用户...【详细内容】
2022-08-22  Tags: 算法  点击:(68)  评论:(0)  加入收藏
0 | 0001100 10100010 10111110 10001001 01011100 00 | 10001 | 1 1001 | 0000 00000000twitter在把存储系统从MySQL迁移到Cassandra的过程中由于Cassandra没有顺序ID生成...【详细内容】
2022-08-20  Tags: 算法  点击:(52)  评论:(0)  加入收藏
雪花算法SnowFlake 算法,是 Twitter 开源的分布式 id 生成算法。其核心思想就是:使用一个 64 bit 的 long 型的数字作为全局唯一 id。在分布式系统中的应用十分广泛,且ID 引入...【详细内容】
2022-08-16  Tags: 算法  点击:(76)  评论:(0)  加入收藏
▌简易百科推荐
作者:小傅哥 博客:https://bugstack.cn沉淀、分享、成长,让自己和他人都能有所收获!一、前言:挂在树上!不知道你经历过HashMap的夺命连环问!为啥,面试官那么喜欢让你聊聊 HashMap?因...【详细内容】
2022-10-10  小傅哥    Tags:红黑树   点击:(8)  评论:(0)  加入收藏
前言在头条创作了一个月左右的时间,收获了50+粉丝,很是开心,我会把数据结构与算法的文章更新到底,第一次看我文章的同仁如果觉得不错的话就关注一下我哦,你的支持就是我创作的动...【详细内容】
2022-10-10  掂掂三生有幸  今日头条  Tags:数据结构   点击:(9)  评论:(0)  加入收藏
一:链表是什么 1、链表是物理存储单元上非连续的、非顺序的存储结构,数据元素的逻辑顺序是通过链表的指针地址实现,有一系列结点(地址)组成,结点可动态的生成。 2、结点包括两个部...【详细内容】
2022-10-07  legendarykk  CSDN  Tags:链表   点击:(9)  评论:(0)  加入收藏
一、什么是递归?自己调用自己,当业务逻辑符合以下三个条件的时候,就可以考虑使用递归来实现。 一个问题可以分解为多个子问题; 当前问题与其子问题除了数据规模不同外,求解思路...【详细内容】
2022-10-07  掂掂三生有幸    Tags:递归算法   点击:(10)  评论:(0)  加入收藏
✨最近有一些粉丝问了我个问题,我平时是怎样学习一门新的技术的,文章开始之前我先来分享一下我的制胜法宝。✨博主学习方法“三刷”官方文档或源码是我高效学习一门新的技能的...【详细内容】
2022-10-04  掂掂三生有幸  今日头条  Tags:链表   点击:(9)  评论:(0)  加入收藏
最经典最常用的排序算法有:冒泡排序、插入排序、选择排序、归并排序、快速排序、计数排序、基数排序和桶排序。这些排序算法可以按照时间复杂度分为三类: O(n^2)&mdash;&mdash...【详细内容】
2022-10-01  掂掂三生有幸  今日头条  Tags:排序算法   点击:(12)  评论:(0)  加入收藏
实现优先级队列最常用的数据结构是堆,堆的常见实现有二叉堆、斐波那契堆、二项堆等。二叉堆堆是一种完全二叉树,我们以小根堆为例,小根堆的性质就是,每个节点都小于其左孩子和右...【详细内容】
2022-09-29  51Testing软件测试网     Tags:数据结构   点击:(8)  评论:(0)  加入收藏
简介布隆过滤器(BloomFilter)是一种用于判断元素是否存在的方式,它的空间成本非常小,速度也很快。但是由于它是基于概率的,因此它存在一定的误判率,它的Contains()操作如果返回tru...【详细内容】
2022-09-13  java保佑我发大财  今日头条  Tags:布隆过滤器   点击:(48)  评论:(0)  加入收藏
1、A GPU accelerated Genetic Algorithm for the Construction of Hadamard Matrices Andras Balogh, Raven Ruiz这篇论文使用遗传算法来构建Hadamard矩阵。 生成随机矩...【详细内容】
2022-09-06  deephub  今日头条  Tags:遗传算法   点击:(59)  评论:(0)  加入收藏
导读:ClickHouse已经成为行业主流且热门的开源引擎。随着业务数据量扩大,场景覆盖变广泛,在复杂query场景下,ClickHouse容易存在查询异常问题,影响业务正常推进。本次主要分享字...【详细内容】
2022-09-05  互联共商     Tags:ClickHouse   点击:(41)  评论:(0)  加入收藏
站内最新
站内热门
站内头条