算法魅力-二分查找实战

目录

前言

算法定义

朴素二分模版

二分查找 

 二分的边界查找

在排序数组中查找元素的第一个和最后一个位置(medium)

 暴力算法

二分查找 

边界查找分析

 山峰数组的峰顶

暴力枚举

二分查找

搜索旋转排序数组中的最小值(medium)

二分查找

结束语


前言

在前面我们学习了双指针,以及其中诞生的分支滑动窗口,接下来我们将探讨其另外一个“兄弟”-二分查找。本质上也是用左右两个指针。

这个算法的前提是我们数据是有序排列的,这里的有序并不只是单纯的有序,有时候根据数据的排列我们可以将数据划分为两个区间,可以简称为二段性,(两段区间是有序的)且根据问题选择合适的二分思路,二分算法有基础的套用也有进阶的实现。

算法定义

二分查找算法(Binary Search Algorithm)是一种在有序数组中查找特定元素的搜索算法。其基本思想是通过不断将搜索区间缩小一半来查找目标值。以下是二分查找算法的步骤:

  1. 首先确定搜索区间的起始位置(left)和结束位置(right)。
  2. 计算中间位置(mid),通常是(left + right) / 2,为了避免溢出也可以写成left + (right - left) / 2。有时候也写成 left + (right - left+1) / 2,两者区别就是在偶数个数据时,一个是取左边,一个是取靠中间右边。可以理解成向下或者向上取整。
  3. 比较中间位置的元素与目标值:
    • 如果中间位置的元素等于目标值,则搜索成功,返回中间位置的索引。
    • 如果中间位置的元素小于目标值,则将搜索区间的起始位置设置为mid + 1,因为目标值必定在右侧区间。
    • 如果中间位置的元素大于目标值,则将搜索区间的结束位置设置为mid - 1,因为目标值必定在左侧区间。
  4. 重复步骤2和3,直到找到目标值或者搜索区间为空(即left > right)。

如果整个数组中没有找到目标值,则返回一个特殊值(如-1)表示未找到。

朴素二分模版

#include <vector>int binarySearch(const std::vector<int>& nums, int target) {int left = 0, right = nums.size() - 1;while (left <= right) {int mid = left + (right - left) / 2;if (nums[mid] == target) {return mid; // 找到目标值,返回索引} else if (nums[mid] < target) {left = mid + 1; // 在右侧区间继续查找} else {right = mid - 1; // 在左侧区间继续查找}}return -1; // 未找到目标值
}

二分查找 

704. 二分查找 - 力扣(LeetCode)

cf07d79c655d4c23b5560986d0b12ebe.png

本题可以通过暴力枚举,通过将数组的数据与目标值进行比较,相等就返回下标,不存在就返回-1.

本题也可以直接就二分查找,就像题目标题一样。

class Solution {
public:int search(vector<int>& nums, int target) {int left=0,right=nums.size()-1;while(left<=right){int mid=left+(right-left)/2;if(nums[mid]<target)left=mid+1;else if(nums[mid]>target)right=mid-1;elsereturn mid;}return -1;}};

 二分的边界查找

有效利用数据的二段性

下面我们将通过一道题来引入进阶的二分。

在排序数组中查找元素的第一个和最后一个位置(medium)



34. 在排序数组中查找元素的第一个和最后一个位置 - 力扣(LeetCode)

8ce424914a93402fbc28050cf6e0a3ab.png

 暴力算法

这道题我们同样可以通过遍历数据来求得左右位置,一个从左边开始查找,一个从右边开始查找,相等就保存并返回到数据中,代码实现也很简单。

class Solution {
public:vector<int> searchRange(vector<int>& nums, int target) {int left=-1,right=-1;int n=nums.size();for(int i=0;i<n;i++){if(nums[i]==target){left=i;break;}}for(int j=n-1;j>=0;j--){if(nums[j]==target){right=j;break;}}return{left,right};}
};

但是这道题让我们设置O(logn)的时间复杂度,同样是查找,故我们可以采用二分查找的思路。

只不过这里要左右两个值,理所当然采用两次二分查找,本质上这道题就是进行左右边界查找。

二分查找 
class Solution {
public:vector<int> searchRange(vector<int>& nums, int target) {if(nums.size()==0)return{-1,-1};int left=0,right=nums.size()-1;while(left<right){int mid=left+(right-left)/2;if(nums[mid]<target)left=mid+1;elseright=mid;}int begin=left;if(nums[left]!=target)return{-1,-1};right=nums.size()-1;while(left<right){int mid=left+(right-left+1)/2;if(nums[mid]>target)right=mid-1;elseleft=mid;}return{begin,right};}
};

边界查找分析

方便叙述,用 x 表示该元素, resLeft 表示左边界, resRight 表示右边界。

左边界查找

6e733b7dece74f8996320f09d867e7bc.png

寻找左边界思路
左边区间 [left, resLeft - 1] 都是小于 x 的;
右边区间(包括左边界) [resLeft, right] 都是大于等于 x 的;
因此,关于 mid 的落点,我们可以分为下面两种情况:
当mid 落在 [left, resLeft - 1] 区间的时候,也就是 arr[mid] < target 。说明 [left, mid] 都是可以舍去的,此时更新 left 到 mid + 1 的位置, 继续在 [mid + 1, right] 上寻找左边界;
当 mid 落在 [resLeft, right] 的区间的时候,也就是 arr[mid] >= target 。
说明 [mid + 1, right] (因为 mid 可能是最终结果,不能舍去)是可以舍去的,此时更新 right 到 mid 的位置,继续在 [left, mid] 上寻找左边界;

注意:这面找中间元素需要向下取整。

因为后续移动左右指针的时候:
左指针: left = mid + 1 ,是会向后移动的,因此区间是会缩小的;
右指针: right = mid ,可能会原地踏步(比如:如果向上取整的话,如果剩下 1,2 两个元
素, left == 1 , right == 2 , mid == 2 。更新区间之后, left,right,mid 的
值没有改变,就会陷入死循环)。
因此一定要注意,当 right = mid 的时候,要向下取整。

6b33fd362b794cf79bbe38aaf806098f.png

寻找右边界思路
用 resRight 表示右边界;
注意到右边界的特点: 左边区间 (包括右边界) [left, resRight] 都是小于等于 x 的;
右边区间 [resRight+ 1, right] 都是大于 x 的;
关于 mid 的落点,可以分为下面两种情况:
当mid 落在 [left, resRight] 区间的时候,说明 [left, mid - 1] ( mid 不可以舍去,因为有可能是最终结果) 都是可以舍去的,此时更新 left 到 mid
的位置;
当 mid 落在 [resRight+ 1, right] 的区间的时候,说明 [mid, right] 内的元素是可以舍去的,此时更新 right 到 mid - 1 的位置;
注意:这里找中间元素需要向上取整。
因为后续移动左右指针的时候:
左指针: left = mid ,可能会原地踏步(比如:如果向下取整的话,如果剩下 1,2 两个元
素, left == 1, right == 2,mid == 1 。更新区间之后, left,right,mid 的值没有改变,就会陷入死循环)。
右指针: right = mid - 1 ,是会向前移动的,因此区间是会缩小的;

 综上所述:

1f6c491c4c1d46e5bce4827ff873da2f.png

当选择两段式的模板时:
在求 mid 的时候,只有 right - 1 的情况下,才会向上取整(也就是 +1 取中间数)

 山峰数组的峰顶



852. 山脉数组的峰顶索引 - 力扣(LeetCode)

12091815b21e43798f4ab6e910f61d92.png

暴力枚举
峰顶的特点:⽐两侧的元素都要⼤。
因此,我们可以遍历数组内的每⼀个元素,找到某⼀个元素⽐两边的元素⼤即可。
class Solution {
public:int peakIndexInMountainArray(vector<int>& arr) {int n = arr.size();// 遍历数组内每⼀个元素,直到找到峰顶for (int i = 1; i < n - 1; i++) // 峰顶满⾜的条件if (arr[i] > arr[i - 1] && arr[i] > arr[i + 1])return i; // 为了处理 oj 需要控制所有路径都有返回值return -1;}
};
二分查找

通过发现题目发现数据存在二段性,峰值左边数值依次递增,峰值右边依次递减。

算法思路:
峰顶数据特点: arr[i] > arr[i - 1] && arr[i] > arr[i + 1] ;
峰顶左边的数据特点: arr[i] > arr[i - 1] && arr[i] < arr[i + 1] ,也就是呈现上升趋势;
峰顶右边数据的特点: arr[i] < arr[i - 1] && arr[i] > arr[i + 1] ,也就是呈现下降趋势。
因此,根据 mid 位置的信息,可以分为下⾯三种情况:
如果 mid 位置呈现上升趋势,说明我们接下来要在 [mid + 1, right] 区间继续搜索;
如果 mid 位置呈现下降趋势,说明我们接下来要在 [left, mid - 1] 区间搜索;
如果 mid 位置就是⼭峰,直接返回结果。
fbaf660f974f41a89f855328d6446b33.png

因为第一个位置和最后一个位置不可能是峰值,所以left=1,right=arr.size()-2;

class Solution
{
public:int peakIndexInMountainArray(vector<int>& arr) {int left = 1, right = arr.size() - 2;while(left < right){int mid = left + (right - left + 1) / 2;if(arr[mid] > arr[mid - 1]) left = mid;else right = mid - 1;}return left;}
};
.

搜索旋转排序数组中的最小值(medium)



153. 寻找旋转排序数组中的最小值 - 力扣(LeetCode)

296a3261c50f45d08f4e1533014ee4ed.png

 暴力解法就是遍历数据直接找最小值。当然也可以直接sort排序,直接返回数组首元素(哈哈哈,这个方法抽象)

class Solution {
public:int findMin(vector<int>& nums) {sort(nums.begin(),nums.end());return nums[0];}
};
二分查找

我们可以发现翻转后的数组来两段区间都是严格递增的。

3b006f12fc144c6ab868c8b71b75107f.png

其中 C 点就是要求的点。
二分的本质:找到一个判断标准,使得查找区间能够一分为二。
通过图像我们可以发现, [A,B] 区间内的点都是严格大于 D 点的值的, C 点的值是严格小
于 D 点的值的。但是当 [C,D] 区间只有一个元素的时候, C 点的值是可能等于 D 点的值
的。 因此,初始化左右两个指针 left , right : 然后根据 mid 的落点,我们可以这样划分下一次查询的区间:
当 mid 在 [A,B] 区间的时候,也就是 mid 位置的值严格大于 D 点的值,下一次查询区间在 [mid + 1,right] 上;
当 mid 在 [C,D] 区间的时候,也就是 mid 位置的值严格⼩于等于 D 点的值,下次
查询区间在 [left,mid] 上。
当区间长度变成 1 的时候,就是我们要找的结果。
class Solution {
public:int findMin(vector<int>& nums) {int left=0,right=nums.size()-1;int x=nums[right];while(left<right){int mid=left+(right-left)/2;if(nums[mid]>x)left=mid+1;elseright=mid;}return nums[left];}
};

结束语

二分查找的讲解就到此结束啦,各位,相信通过这些题目的讲解大家对二分有了全新的认识和理解,下个算法我们将学习前缀和,欢迎大家来指导交流。

最后感谢各位友友的支持!!!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/472472.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

# 第20章 Cortex-M4-触摸屏

第20章 Cortex-M4-触摸屏 20.1 触摸屏概述 20.1.1 常见的触摸屏分类 电阻式触摸屏、电容式触摸屏、红外式触摸屏、表面声波触摸屏 市场上用的最多的是电阻式触摸屏与电容式触摸屏。红外管式触摸屏多用于投影仪配套设备。 电阻式触摸屏构成&#xff1a;整个屏由均匀电阻构成…

大数据新视界 -- 大数据大厂之 Impala 性能优化:基于数据特征的存储格式选择(上)(19/30)

&#x1f496;&#x1f496;&#x1f496;亲爱的朋友们&#xff0c;热烈欢迎你们来到 青云交的博客&#xff01;能与你们在此邂逅&#xff0c;我满心欢喜&#xff0c;深感无比荣幸。在这个瞬息万变的时代&#xff0c;我们每个人都在苦苦追寻一处能让心灵安然栖息的港湾。而 我的…

Gitcode文件历史记录查看和还原

文件历史记录 文件历史记录用于记录代码文件的更改历史&#xff0c;它允许用户查看文件的不同版本&#xff0c;了解每个版本的修改内容、作者和提交消息。这对于跟踪文件演进、恢复错误更改、审查代码以及了解项目进展都非常有用。 文件历史记录功能提供了以下核心功能&#…

数据结构-二叉树及其遍历

🚀欢迎来到我的【数据结构】专栏🚀 🙋我是小蜗,一名在职牛马。🐒我的博客主页​​​​​​ ➡️ ➡️ 小蜗向前冲的主页🙏🙏欢迎大家的关注,你们的关注是我创作的最大动力🙏🙏🌍前言 本篇文章咱们聊聊数据结构中的树,准确的说因该是只说一说二叉树以及相…

Java集合(Collection+Map)

Java集合&#xff08;CollectionMap&#xff09; 为什么要使用集合&#xff1f;泛型 <>集合框架单列集合CollectionCollection遍历方式List&#xff1a;有序、可重复、有索引ArrayListLinkedListVector&#xff08;已经淘汰&#xff0c;不会再用&#xff09; Set&#xf…

Python学习------第八天

函数 函数的传入参数 掌握函数返回值的作用 掌握函数返回值的定义语法 函数的嵌套调用&#xff1a; 函数的局部变量和全局变量 局部变量的作用&#xff1a;在函数体内部&#xff0c;临时保存数据&#xff0c;即当函数调用完成后&#xff0c;则销毁局部变量。 money 5000000 n…

reduce-scatter:适合分布式计算;Reduce、LayerNorm和Broadcast算子的执行顺序对计算结果的影响,以及它们对资源消耗的影响

目录 Gather Scatter Reduce reduce-scatter:适合分布式计算 Reduce、LayerNorm和Broadcast算子的执行顺序对计算结果的影响,以及它们对资源消耗的影响 计算结果理论正确性 资源消耗方面 Gather 这个也很好理解,就是把多个进程的数据拼凑在一起。 Scatter 不同于Br…

C++- 基于多设计模式下的同步异步日志系统

第一个项目:13万字,带源代码和详细步骤 目录 第一个项目:13万字,带源代码和详细步骤 1. 项目介绍 2. 核心技术 3. 日志系统介绍 3.1 为什么需要⽇志系统 3.2 ⽇志系统技术实现 3.2.1 同步写⽇志 3.2.2 异步写⽇志 4.知识点和单词补充 4.1单词补充 4.2知识点补充…

Node.js GET/POST请求、WEB模块使用介绍 (基础介绍 八)

GET/POST请求 在很多场景中&#xff0c;我们的服务器都需要跟用户的浏览器打交道&#xff0c;如表单提交。 表单提交到服务器一般都使用 GET/POST 请求。 本章节我们将为大家介绍 Node.js GET/POST请求。 获取GET请求内容 由于GET请求直接被嵌入在路径中&#xff0c;URL是…

字节青训-小M的多任务下载器挑战、版本号比较

目录 一、小M的多任务下载器挑战 题目背景 题目内容 数据输入 数据输出 数据与约定 示例1 示例2 解题思路&#xff1a; 问题理解 数据结构选择 算法步骤 最终代码&#xff1a; 运行结果&#xff1a; 二、版本号比较 问题描述 样例 示例 1: 示例 2: 示例 3:…

jenkins用户在执行scp的时候如何做免密登录

一、背景 在jenkins job中执行scp的shell命令&#xff0c;当然不希望每次输入密码&#xff0c;另外处于出于安全考虑&#xff0c;也不建议在scp命令中指定。 所以&#xff0c;我们需要对远程机器进行免密登录。 本文遇到的问题是&#xff0c;在jenkins机器上执行scp已做到了…

Prometheus监控SQL SERVER常用指标和PromQL预警

SQL Server是企业级广泛应用的数据库&#xff0c;通过简单的Prometheus exportor可以很容易地监控它。与所有数据库一样&#xff0c;SQL Server也有许多故障点&#xff0c;例如事务延迟或数据库中连接过多。本文介绍如何使用Prometheus监视SQL Server&#xff0c;包括常用的监控…

HTML5实现俄罗斯方块小游戏

文章目录 1.设计来源1.1 主界面1.2 皮肤风格1.2 游戏中界面1.3 游戏结束界面 2.效果和源码2.1 动态效果2.2 源代码 源码下载 作者&#xff1a;xcLeigh 文章地址&#xff1a;https://blog.csdn.net/weixin_43151418/article/details/143788449 HTML5实现俄罗斯方块小游戏&#x…

从北美火到中国,大数据洞察品牌“STANLEY”的突围之路

保守直筒大头的“硬汉”外形&#xff0c;以百变颜色踩中时尚命脉&#xff0c;与各路大牌“梦幻联动”&#xff0c;不少时尚弄潮儿没能逃过其“真香”诱惑。 这就是今年以来从北美火到中国的STANLEY&#xff0c;在“巨无霸”水杯中突围出属于自己的一条路。 最近STANLEY又整活…

linux逻辑卷练习

目录 知识点&#xff1a; 常用命令 题目&#xff1a; 解题&#xff1a; 1&#xff09;分区 2&#xff09;创建物理卷 3&#xff09;创建卷组 4&#xff09;生成逻辑卷 "要带参数 -n" 5&#xff09;扩容 6&#xff09;格式化(添加文件系统) 7&#xff09;挂…

java版嘎嘎快充汽车单车充电系统源码系统jeecgboot

汽车使用云快充1.6 1.5协议&#xff0c;单车用的铁塔协议 前端uniapp、后端jeecgbootvue2

【Vitepress报错】Error: [vitepress] 8 dead link(s) found.

原因 VitePress 在编译时&#xff0c;发现 死链接(dead links) 会构建失败&#xff01;具体在哪我也找不到… 解决方案 如图第一行蓝色提示信息&#xff0c;设置 Vitepress 属性 ignoredeadlinks 为 true 可忽略报错。 .vuepress/config.js export default defineConfig(…

《Django 5 By Example》阅读笔记:p105-p164

《Django 5 By Example》学习第5天&#xff0c;p105-p164总结&#xff0c;总计60页。 一、技术总结 1.文章标签功能 Django自带django-taggit。 2.自定义template tags 3.roadmap功能 4.RSS功能 5.full-text搜索功能 这里使用的是Postgresql,使用pip install psycopg安…

awk(常用)

这个有点难 O.o 一、awk # 语法 awk 参数 模式 {动作} 文件# 第一列&#xff0c;包含p的 $1~"p" # 第一列&#xff0c;不包含p的 $1!~"p" # 开始时干嘛&#xff0c;结束时干嘛 awk BEGIN{开始时做的事}END{结束时做的事}{print $0} 文件 1、内置变量&…

数据结构—栈和队列

目录 1.栈底层结构的选择 2.栈的实现 3.栈 3.1入栈 3.2出栈 3.3栈顶删除 4.队列 4.1队列介绍 4.2队列初始化 4.3入队列 4.4队头删除 1.栈底层结构的选择 栈是一种数据结构 具有“后进先出的”的特点 现在面临的两种选择&#xff0c;一种是顺序表&#xff0c;另一种…