二分查找法搜寻元素 Leetcode35, Leetcode69

news/2025/1/16 4:42:05/

二分查找法 ( Binary Search ) 常用于在 有序数组按值查找 某个元素,返回其索引。二分查找可以极大提高搜索效率,其时间复杂度是 O(log N) (N 为数组长度)。我最近在 Leetcode 做了一些二分查找的题目,花了一些时间才弄明白解决问题的思路,在这里归纳总结一下,希望也能帮你更快地梳理思路,欢迎交流讨论呀!

一、标准二分查找 :查找某个元素

Leetcode 704. 二分查找 是一道标准的二分查找题,要在一个升序整型数组 nums (无重复元素)中寻找目标值 target。如果 target 存在,就返回索引,否则返回 -1。

二分查找的要点是每次把搜索范围缩小一半。具体做法就是每次把当前区间 [ left, right ] 中间点位置的值 nums[middle] 与 target 对比,有三种情况:

  • 如果 nums[middle] 等于 target,这就是要找的元素。
  • 如果 nums[middle] 小于 target,接下来只在大于中间点的区间 [ middle + 1, right ] 搜索。
  • 如果 nums[middle] 大于 target,接下来只要在小于中间点的区间 [ left, middle - 1 ] 搜索。

如果搜索结束,没有找到符合条件的元素,返回 -1。

对应的 Python 代码如下:

left, right = 0, len(nums)-1while left <= right:# 这里求 middle的表达式与 (left + right)//2 相同# 这样写是为了防止数值太大时,相加运算溢出   middle = left + (right - left) // 2if nums[middle] == target:return middleelif nums[middle] < target:left = middle + 1else:right = middle - 1return -1

这道问题对于没找到值为 target 的元素的情况,只要返回 -1 即可。如果增加一点难度,当没有在数组中找到值为 target 的元素时,要返回一个与它最接近的元素,那应该怎么办呢?接下来我们就来分析解决这一类问题。

二、查找取值与 target 最接近的元素

这类问题增加了 对于数组中没有值为 target 的元素时的处理,此时依然要返回一个索引,就是与 target 值最接近的元素的索引。

情况 1 :查找 >= target 、且与 target 值最接近的元素,例如, Leetcode 35. 搜索插入位置 ,当数组中没有找到值为 target 的元素时, 要求返回把 target 按顺序插入数组时的位置。例如,输入 nums = [ 1, 3, 5, 7 ], target = 2,输出为 1,也就是元素 3 的索引。在输入 nums 中,元素 3 是大于 target 的元素中、最接近 target 的元素。

情况 2 :查找 <= target 、且与 target 值最接近的元素,例如, Leetcode 69. x 的平方根 ,求一个非负整数 x 的平方根,要求返回结果是整型。如果遇到平方根不是整数的情况呢?只取整数部分。例如,输入 x = 8,输出为 2。8 的平方根也就是 target 值,是小数 2.82842…。2 是小于 target 的元素中、最接近 target 的元素。

方法一、标准二分查找法 while left <= right

思路来源:《算法》(Robert Sedgewick, Kevin Wayne 著)第 3.1节

这个方法应用标准二分查找法,只需改动 while 循环之后的语句 return -1

  • 如果查找的是 >= target 的元素,改为 return left
  • 如果查找的是 <= target 的元素,改为 return right

为什么呢?接下来我们来分析一下。当数组中没有值为 target 的元素时,因为 while 循环的条件是 left <= right,最后一次循环时搜寻区间有一个或两个元素,right = left 或 left +1,这两种情况时都有 middle = left。

情况一、返回 > target、最接近 target 的元素索引,例如:Leetcode 35. 搜索插入位置 。

  • 如果最后一次 while 循环时 nums[middle] > target,元素应该插入的位置是 middle,而循环结束时 left = middle。
  • 如果最后一次 while 循环时 nums[middle] < target,元素应该插入的位置是 middle + 1,而循环结束时 left = middle +1。

因此,只需把标准二分查找代码中 while 循环之后的语句由 return -1 改为 return left 即可。

情况二、返回 < target、最接近 target 的元素索引,例如:Leetcode 69. x 的平方根 。

  • 如果最后一次 while 循环时 nums[middle] > target,元素应该插入的位置是 middle - 1 ,而循环结束时 right = middle -1。

  • 如果最后一次 while 循环时 nums[middle] < target,元素应该插入的位置是 middle。循环开始时只有 right = left = middle(如果 while 循环开始时 right = left + 1,而 nums[middle] < target,还会进入下一次循环,因此排除这种情况。),结束时 right = middle。

因此, 只需把标准二分查找代码中 while 循环之后的语句由 return -1 改为 return right 即可。

方法二、另一种二分查找法 while left < right

思路来源:01.二分查找知识 | 算法通关手册

方法一,也就是标准二分查找法,是通过不断缩小搜索范围来查找某个元素。但是我们解决这一类型问题时发现,target 的取值可能是介于两个元素中间,虽然 nums[middle] 不等于 target,也许它就是最接近 target 取值的元素,比如对于搜索插入位置的情况,当输入为 nums = [ 1, 3, 5, 7 ] , target = 2, middle = 3 时。因此,我们保留 middle 位置元素进入下一次搜寻,这就是方法二的思路。

情况一、返回 > target、最接近 target 的元素索引,例如:Leetcode 35. 搜索插入位置

这种情况下,当 nums[middle] > target 时,middle 位置有可能是我们要找的插入位置,下一次搜寻区间应该包含 middle。因此,此时 right = middle

与方法一不同,这里的 while 循环条件为 left < right,因此循环终止时有 left = right,这样搜寻区间还有一个元素。这就是 >= target、最接近 target 的元素,最终返回它的索引( left 或 right 都可以,两者相同)。

Python 代码如下:

# 如果target不大于nums的第一个元素,直接返回索引0
if target <= nums[0]:return 0
# 如果target大于nums的最后一个元素,直接返回数组长度
if target > nums[len(nums)-1]:return len(nums)left, right = 1, len(nums)-1
while left < right:middle = left + (right - left) // 2if nums[middle] == target:return middleelif nums[middle] < target:left = middle + 1else:right = middlereturn left

情况二、返回 < target、最接近 target 的元素索引,例如:Leetcode 69. x 的平方根

这种情况下,当 nums[middle] < target 时,middle 位置有可能是我们要找的,下一次搜寻区间应该包含 middle。因此,此时 left = middle

这里有一个小细节需要注意,因为 while 循环条件为 left < right,最后一次循环时搜寻区间有两个元素,当 left = right 时循环结束。但是我们求 middle 并不是精确的平均值,而是向下取整,这导致当搜寻区间只有两个元素时,middle 始终等于 left。 这样,当 nums[middle] < target 时,left = middle,下一次循环 middle = (left + right) /2 = left ,没有更新搜寻区间,循环无法停止。怎么办呢?

可以在求均值时对 middle 向上取整,比如 middle = left + (right - left) // 2 + 1middle = left + (right - left + 1) // 2

Python 代码如下:

# x=0或1时,直接返回结果
if x <= 1:return xleft, right = 1, x
while left < right:middle = left + (right - left) // 2 + 1if middle ** 2 == x:return middleelif middle ** 2 > x:right = middle - 1else:left =  middlereturn left

本文对您有帮助的话,请点赞支持一下吧,谢谢!

关注我 宁萌Julie,互相学习,多多交流呀!

参考

  1. 《算法》(Robert Sedgewick, Kevin Wayne 著)

  2. https://algo.itcharge.cn/01.Array/03.Array-Binary-Search/01.Array-Binary-Search/


http://www.ppmy.cn/news/282640.html

相关文章

联合索引详解

联合索引详解 前言 在数据库中&#xff0c;索引是一种重要的数据结构&#xff0c;用于提高查询效率。而联合索引是一种特殊的索引类型&#xff0c;它可以同时索引多个列。联合索引在实际应用中非常常见&#xff0c;但是很多人对它的理解还不够深入。本文将从联合索引的定义、…

深入理解深度学习——注意力机制(Attention Mechanism):注意力汇聚与Nadaraya-Watson 核回归

分类目录&#xff1a;《深入理解深度学习》总目录 《深入理解深度学习——注意力机制&#xff08;Attention Mechanism&#xff09;&#xff1a;基础知识》介绍了框架下的注意力机制的主要成分&#xff1a; 查询&#xff08;自主提示&#xff09;和键&#xff08;非自主提示&am…

机器学习 | 支持向量机SVM | 概念了解向

概念了解向&#xff0c;参考视频&#xff1a; 【小萌五分钟】机器学习 | 支持向量机 SVM &#x1f4da;最大间隔分类器 如下图有两种不同颜色的点。我需要一个分类器告诉我&#xff0c;假设在下图中新加入一个点&#xff0c;应该将它分类至红点还是蓝点。考虑加入一条决策边界…

Shell脚本:expect脚本免交互

Shell脚本&#xff1a;expect脚本免交互 expect脚本免交互 一、免交互基本概述&#xff1a;1.交互与免交互的区别&#xff1a;2.格式&#xff1a;3.通过read实现免交互&#xff1a;4.通过cat实现查看和重定向&#xff1a;5.变量替换&#xff1a; 二、expect安装&#xff1a;1.…

NT98520/NT98525(典型)/NT98528参数对比

联咏Novatek NT98525是一款高集成度的SoC&#xff0c;具有高图像质量、低比特率、低功耗的特点&#xff0c;目标是4Mp到5Mp边缘计算IP摄像机的应用。SoC集成 ARM Cortex A9 CPU核&#xff0c;新一代ISP&#xff0c;H.265/H.264视频压缩编解码器&#xff0c;高性能硬件DLA模块、…

三星note5 android 8.0,好!这些三星设备可以升级到Android 8.0,S6,缺少Note5

1月25日&#xff0c;三星正式宣布将于北京时间2月26日上午1:00举行新产品发布会&#xff0c;届时将发布Galaxy S9和S9 . 毫无疑问&#xff0c;这两个旗舰将预装基于Android 8.0定制的Samsung Experience 9. 但是&#xff0c;越来越多的用户对他们的旧三星设备是否可以更新到最新…

解锁三星bl锁有几种方法_三星Note5解锁教程_三星Note5 CROM解BL锁的方法

来说说咱们的三星Note5手机如何进行解锁吧&#xff0c;因为手机不解锁的话&#xff0c;很多操作都进行不了&#xff0c;这也是现在最新的机子普遍存在的问题&#xff0c;很多机友喜欢刷自制的rom&#xff0c;并且需要刷第三方的recovery&#xff0c;这个时候咱们的手机必须先要…

note5

攻防世界 Reverse easyre-xctf 题目描述:龟缩起来&#xff0c;保护自己//猜测可能与逆向中的脱壳&#xff0c;加壳有关 壳就是很多生物表面覆盖包裹的一层东西&#xff0c;有加壳就脱壳 打开压缩包&#xff0c;运行出现以下窗口&#xff0c;感觉应该和upx有关&#xff0c;打开…