二分查找算法介绍及使用

目录

[1. 什么是二分查找](#1. 什么是二分查找)

[2. 二分查找的逻辑](#2. 二分查找的逻辑)

[什么是二分查找的 "二段性"?](#什么是二分查找的 “二段性”?)

[3. 例题讲解演示](#3. 例题讲解演示)

[3.1 LeetCode704. 二分查找](#3.1 LeetCode704. 二分查找)

[3.2 LeetCode852. 山脉数组的峰顶索引](#3.2 LeetCode852. 山脉数组的峰顶索引)

[3.3 LeetCode69. x 的平方根](#3.3 LeetCode69. x 的平方根)

[3.4 LeetCode34. 在排序数组中查找元素的第一个和最后一个位置](#3.4 LeetCode34. 在排序数组中查找元素的第一个和最后一个位置)


今天我们来聊一聊二分查找算法。

1. 什么是二分查找

分查找(Binary Search)是一种针对有序集合 的高效查找算法,核心思想是通过不断将查找范围减半来快速定位目标元素。其前提是集合必须按某种规则排序,每次通过比较中间元素与目标值的大小,缩小查找范围,直到找到目标或范围为空。

时间复杂度为 O(log n)(n 为集合长度),远优于线性查找的 O (n),适用于静态有序数据的快速查询。

二分查找的效率很高,因为他每次都可以直接排除掉一半的数据。

2. 二分查找的逻辑

二分查找的本质就是通过找到数组的二段性,然后每次筛选一半的方式来快速查找。

我们要了解二分查找的逻辑,那么我们就先要了解什么是二段性。

什么是二分查找的 "二段性"?

"二段性" 是二分查找的核心本质,指待查找的集合可以被某个条件划分为 "满足条件" 和 "不满足条件" 的两个连续区间,且这两个区间具有明确的边界。

简单来说:

  • 整个集合被分为两部分,前半部分(或左半部分)全部满足某个条件,后半部分(或右半部分)全部不满足(反之亦然)。
  • 二分查找的过程,本质上是通过不断判断中间元素属于哪一段,来缩小范围,最终找到两段的边界(或目标元素)。

给各位句几个简单的例子,我们看下面这个图,如果我们要查找的是3,那么它的二段性就是大于3的数和小于3的数。我们通过不断筛选,最后找到3。

我们看下面这个图,那么我们该如何找到这组数里面最大的。如果只知道二分查找可以使用到有序数组的话,我们在下面这个图里面是想不到使用二分查找的。那么我们该怎么使用二分查找呢,或者说二段性该如何理解呢?很简单,它的二段性就是上升期的数和下降期的数。这样我们就可以找到里面最大的。

3. 例题讲解演示

接下来我们通过讲解几道例题的方式来帮助大家理解二分查找。

3.1 LeetCode704. 二分查找

我们看下面这个图片,这个的话就是很普通的二分查找例题。就是要求我们从一个有序的数组里面找到要求的数字,如果那个数字不在数组里面的话,那就返回-1。

它的二段性就是比它大的值和比它小的值。

我们看下面的代码,就是先设置一个l和一个r,分别指向数组的最左边和最右边,接着再根据l+(r-l)/2计算出mid,也就是中间位置的下标。接着我们拿这个位置的下标对应的值去和目标值作比较,如果相等就直接结束,否则就把l或者r更新为mid。

这个while循环里面的条件很好理解,如果l>r了那么就说明没有找到,所以直接返回-1。

cpp 复制代码
class Solution {
public:
    int search(vector<int>& nums, int t) {
        int sz=nums.size();
        int l=0;
        int r=sz-1;
        while(l<=r)
        {
            int mid=l+(r-l)/2;
            if(nums[mid]>t)
                r=mid-1;
            else if(nums[mid]<t)
                l=mid+1;
            else
            {
                return mid;
            }
        }
        return ;
    }
};

3.2 LeetCode852. 山脉数组的峰顶索引

我们接着看下面这一道题,这道题就是我上面说的那种找最大值。

它的二段性就是上升期的数和下降期的数。

我们看下面这个代码,其实很多地方也是差不多的,唯一的区别就是把两个条件判断给修改一下。

因为要判断是上升期的数还是下降期的数,所以我们要和和周围的数去比较。

cpp 复制代码
class Solution {
public:
    int peakIndexInMountainArray(vector<int>& arr) {
        int sz=arr.size();
        int l=0;
        int r=sz-1;
        while(l<r)
        {
            int m=l+(r-l+1)/2;
            if(arr[m-1]<arr[m])
                l=m;
            else if(arr[m-1]>arr[m])
                r=m-1;
        }
        return r;
    }
};

3.3 LeetCode69. x 的平方根

我们来看下面这道题,这道题同样也可以使用二分查找算法。题目意思很简单,要求我们找出一个数的算术平方根。

它的二段性就是和自己相乘后比x要大的数和比x要小的数。

因为算术平方根就是一个数开根号后的结果,所以我们这边直接通过设立边界,然后查找哪个数的值和自己相乘的结果等于x。

如果没有拿到我们想要的也没有事,因为当l>r时的那个值就是去除小数后的结果。

cpp 复制代码
class Solution {
public:
    int mySqrt(int x) {
        int l=0;
        int r=x;
        if(x<2)
            return x;
        while(l<=r)
        {
            long long mid=l+(r-l)/2;
            if(mid*mid>x)
                r=mid-1;
            else if(mid*mid<x)
                l=mid+1;
            else
                return mid;
        }
        return r;
    }
};

3.4 LeetCode34. 在排序数组中查找元素的第一个和最后一个位置

这道题的目的是让各位了解到二分查找使用上的广泛性。

这道题稍微有一点难,题目意思就是说给一个值,然后要求我们找到数组里面这个数出现的起始位置和结尾位置。如果数组里面没有这个数的话就直接返回[-1,-1]。

这道题的二段性比较特殊,放在下面的代码解析那里说。

这道题我们可以理解为使用两次二段性,首先我们通过二段性查找t值在数组中的起始位置,也就是左边界,二段性为左边全是比要求小的值,右边为等于或大于要求的值。接着我们查找结尾位置,也就是右边界,二段性为右边全是比要求大的值,右边为等于或小于要求的值。这样我们就找到了左右边界,这样我们就做完了这道题。

cpp 复制代码
class Solution {
public:
    vector<int> searchRange(vector<int>& nums, int t) {
        int sz=nums.size();
        int l=0;
        int r=sz-1;
        int begin=0;
        int end=0;
        if(sz==0)
            return {-1,-1};
        //找左
        while(l<r)
        {
            int m=l+(r-l)/2;
            if(nums[m]>=t)
                r=m;
            else if(nums[m]<t)
                l=m+1;
        }
        if(nums[l]!=t)
            return {-1,-1};
        else
            begin=l;
        //找右
        l=0;
        r=sz-1;
        while(l<r)
        {
            int m=l+(r-l+1)/2;
            if(nums[m]>t)
                r=m-1;
            else if(nums[m]<=t)
                l=m;
        }
        end=r;
        return {begin,end};
    }
};
相关推荐
45288655上山打老虎1 分钟前
【智能指针】
开发语言·c++·算法
稚辉君.MCA_P8_Java5 分钟前
Gemini永久会员 go数组中最大异或值
数据结构·后端·算法·golang·哈希算法
会员果汁15 分钟前
双向链式队列-C语言
c语言·数据结构
Michelle802316 分钟前
机器学习实战操作手册
人工智能·算法·机器学习
HaiLang_IT27 分钟前
【目标检测】基于卷积神经网络的轨道部件(扣件、轨枕、钢轨)缺陷检测算法研究
算法·目标检测·cnn
草莓熊Lotso28 分钟前
《算法闯关指南:优选算法--前缀和》--31.连续数组,32.矩阵区域和
c++·线性代数·算法·矩阵
csuzhucong29 分钟前
斜转魔方、斜转扭曲魔方
前端·c++·算法
C语言不精1 小时前
c语言-优雅的多级菜单设计与实现
c语言·开发语言·算法
AI科技星1 小时前
张祥前统一场论:引力场与磁矢势的关联,反引力场生成及拉格朗日点解析(网友问题解答)
开发语言·数据结构·经验分享·线性代数·算法
C雨后彩虹1 小时前
最少交换次数
java·数据结构·算法·华为·面试