算法魅力-二分查找实战

目录

前言

算法定义

朴素二分模版

二分查找 

 二分的边界查找

在排序数组中查找元素的第一个和最后一个位置(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;
            else
            return 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;
            else
            right=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;
            else
            left=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;
        else
        right=mid;
       }
       return nums[left];
    }
};

结束语

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

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

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Linux第四讲:Git gdb

Linux第四讲&#xff1a;Git && gdb 1.版本控制器Git1.1理解版本控制1.2理解协作开发1.3Git的历史1.4Git的操作1.4.1仓库创建解释、仓库克隆操作1.4.2本地文件操作三板斧1.4.3文件推送详细问题 2.调试器 -- gdb/cgdb使用2.1调试的本质是什么2.2watch命令2.3set var命令…

海底捞点单

单点锅底推荐&#xff1a; 番茄锅底通31 牛油麻辣通44 清汤麻辣备44 菌汤锅底通31 小吃&主食&#xff1a; 捞派捞面一黄金小馒头一茴香小油条 红糖枇杷一小酥肉 DIY锅底推荐&#xff1a; 1.寿喜锅&#xff1a;海鲜味酱4勺陈醋1勺蚝油2勺盐适量白糖7勺 芹菜1勺 2.麻辣锅底…

PNG图片批量压缩exe工具+功能纯净+不改变原始尺寸

小编最近有一篇png图片要批量压缩&#xff0c;大小都在5MB之上&#xff0c;在网上找了半天要么就是有广告&#xff0c;要么就是有毒&#xff0c;要么就是功能复杂&#xff0c;整的我心烦意乱。 于是我自己用python写了一个纯净工具&#xff0c;只能压缩png图片&#xff0c;没任…

达梦8数据库适配ORACLE的8个参数

目录 1、概述 1.1 概述 1.2 实验环境 2、参数简介 3、实验部分 3.1 参数BLANK_PAD_MODE 3.2 参数COMPATIBLE_MODE 3.3 参数ORDER_BY_NULLS_FLAG 3.4 参数DATETIME_FMT_MODE 3.5 参数PL_SQLCODE_COMPATIBLE 3.6 参数CALC_AS_DECIMAL 3.7 参数ENABLE_PL_SYNONYM 3.8…

如何低成本、零代码开发、5分钟内打造一个企业AI智能客服?

传统客服因员工效率低、时段需求波动大、数据管理费时费力等管理难题&#xff0c;导致难以满足用户需求&#xff0c;无法深入挖掘客服数据价值&#xff0c;造成客源流失。而智能体搭建的“智能客服”能借助大模型和知识库知识&#xff0c;助力实现数字化运营&#xff0c;破解企…

CC1链学习记录

&#x1f338; 前言 上篇文章学习记录了URLDNS链&#xff0c;接下来学习一下Common-Colections利用链。 &#x1f338; 相关介绍 Common-Colections是Apache软件基金会的项目&#xff0c;对Java标准的Collections API提供了很好的补充&#xff0c;在其基础上对常用的数据结构…

【启明智显分享】5G CPE为什么适合应用在连锁店中?

连锁门店需要5G CPE来满足其日益增长的网络需求&#xff0c;提升整体运营效率和竞争力。那么为什么5G CPE适合连锁店应用呢&#xff0c;小编为此做了整理&#xff0c;主要是基于以下几个方面的原因&#xff1a; 一、高效稳定的网络连接 1、高速数据传输&#xff1a; 5G CPE能…

怎么禁止文件外发?企业如何禁止文件外发,教你6种方法,综合运用效果加倍

在当今数字化的商业环境中&#xff0c;企业内部文件承载着大量关键信息&#xff0c;犹如企业的命脉。这些文件可能包含着核心技术机密、客户资料、未公开的战略规划以及敏感的财务数据等&#xff0c;它们是企业在激烈市场竞争中立足的重要资产。然而&#xff0c;随着信息传播途…

FPGA学习笔记#8 Vitis HLS优化总结和案例程序的优化

本笔记使用的Vitis HLS版本为2022.2&#xff0c;在windows11下运行&#xff0c;仿真part为xcku15p_CIV-ffva1156-2LV-e&#xff0c;主要根据教程&#xff1a;跟Xilinx SAE 学HLS系列视频讲座-高亚军进行学习 学习笔记&#xff1a;《FPGA学习笔记》索引 FPGA学习笔记#1 HLS简介及…

替代CX8835车充5V/3.4A恒流恒压降压转换器内置MOS

概述&#xff1a;PIN对PIN替代CX8835 PC3313是宽输入电压&#xff0c;高效率有源CC降压DC/DC转换器&#xff0c;可以在CV&#xff08;恒压&#xff09;或CC模式下运行&#xff08;恒流&#xff09;输出模式。PC3313在5V输出时提供高达典型3.4A的电流限制带有18mΩ电流感测电阻…

2024年8个最佳在线websocket调试工具选择

精选了 8 款功能强大且易于使用的 WebSocket 测试工具&#xff1a; 工具名称支持的系统是否免费ApifoxWindows, Mac, Linux是WebSocket KingWindows, Mac, Linux是PostmanWindows, Mac, Linux是Socket.IO Test ClientWindows, Mac, Linux是InsomniaWindows, Mac, Linux是Wires…

ML 系列:第 21 节 — 离散概率分布(二项分布)

一、说明 二项分布描述了在固定数量的独立伯努利试验中一定数量的成功的概率&#xff0c;其中每个试验只有两种可能的结果&#xff08;通常标记为成功和失败&#xff09;。 二、探讨伯努利模型 例如&#xff0c;假设您正在抛一枚公平的硬币 &#xff08;其中正面成功&#xff…

2024开发者浏览器必备扩展,不允许还有人不知道~

在开发过程中&#xff0c;优秀的扩展工具能够极大提升我们的工作效率&#xff0c;简化工作流程&#xff0c;并使得在浏览器中的开发和调试变得更加便捷。 根据市场占比&#xff0c;Chrome、Safari、Edge、Firefox、Opera 是前五大浏览器&#xff0c;其中Chrome浏览器占据了领先…

安装paddle

网址&#xff1a;飞桨PaddlePaddle-源于产业实践的开源深度学习平台 或者找对应python和cuda版本的paddle下载后安装&#xff1a; https://www.paddlepaddle.org.cn/whl/linux/mkl/avx/stable.html 你想要安装paddlepaddle - gpu2.6.1.post112版本。在你提供的文件列表中&am…

(六)Spark大数据开发实战:豆瓣电影数据处理与分析(scala版)

目录 一、Spark 二、数据介绍 三、Spark大数据开发实战(Scala) 1、数据文件上传HDFS 2、导入模块及数据 3、数据统计与分析 ①、计算演员参演电影数 ②、依次罗列电影番位前十的演员 ③、按照番位计算演员参演电影数 ④、求每位演员所有参演电影中的最早、最晚上映…

‘nodemon‘ 不是内部或外部命令,也不是可运行的程序

解决方法&#xff1a;使用 npx 临时运行 nodemon 如果你不想全局安装 nodemon&#xff0c;你可以使用 npx&#xff08;npm 5.2 及以上版本自带&#xff09;来临时运行 nodemon&#xff1a; npx nodemon server.jsnodemon正常配置 要在开发过程中实现每次修改 Node.js 代码后…

Docker 的安装与使用

Docker 的安装 Docker 是一个开源的商业产品&#xff0c;有两个版本&#xff1a;社区版&#xff08;Community Edition&#xff0c;缩写为 CE&#xff09;和企业版&#xff08;Enterprise Edition&#xff0c;缩写为 EE&#xff09;。 Docker CE 的安装请参考官方文档&#xf…

单相锁相环,原理与Matlab实现

单相锁相环基本原理 单相锁相环的基本原理图如下所示&#xff0c; u α u_\alpha uα​ u β u_\beta uβ​经Park变换、PI控制实现对角频率 ω \omega ω和角度 θ \theta θ的估算。不同锁相环方案之间的差异&#xff0c;主要表现在正交电压 u β u_\beta uβ​的生成&#x…

LLMs之PDF:zeroX(一款PDF到Markdown 的视觉模型转换工具)的简介、安装和使用方法、案例应用之详细攻略

LLMs之PDF&#xff1a;zeroX(一款PDF到Markdown 的视觉模型转换工具)的简介、安装和使用方法、案例应用之详细攻略 目录 zeroX的简介 1、支持的文件类型 zeroX的安装和使用方法 T1、Node.js 版本&#xff1a; 安装 使用方法 使用文件 URL&#xff1a; 使用本地路径&…

Redis集群模式之Redis Sentinel vs. Redis Cluster

在分布式系统环境中&#xff0c;Redis以其高性能、低延迟和丰富的数据结构而广受青睐。随着数据量的增长和访问需求的增加&#xff0c;单一Redis实例往往难以满足高可用性和扩展性的要求。为此&#xff0c;Redis提供了两种主要的集群模式&#xff1a;Redis Sentinel和Redis Clu…