前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >14天算法入门第一天:二分查找算法,长文详解,包教包会!

14天算法入门第一天:二分查找算法,长文详解,包教包会!

作者头像
川川菜鸟
发布2021-10-19 10:32:49
2870
发布2021-10-19 10:32:49
举报

一、算法详细讲解

1.0 前言介绍

讲解已经非常详细,尽量是让小白都能学会,因此如果你觉得自己算法并不是很好,或者没有基础,那我希望你一定要认真看我写的每一句话,同时要学会自己思考,不然对你也收获不大。

1.1二分查找介绍

二分查找也叫做折半查找。查找也是有特殊情况的,比如数列本身是有序的。这个有序数列是怎么产生的呢?有时它可能本身就是有序的,也有可能是我们通过之前所学的排序算法得到的。

1.2二分查找条件

由上面的讲解可以知道,二分查找有两个要:,一个是数列有序,另一个是数列使用顺序存储结构(比如数组)

二、 原理及实现

二分查找的实现原理非常简单,首先要有一个有序的列表。但是如果没有,则该怎么办?可以使用排序算法进行排序。 以升序数列为例,比较一个元素与数列中的中间位置的元素的大小,如果比中间位置的元素大,则继续在后半部分的数列中进行二分查找;如果比中间位置的元素小,则在数列的前半部分进行比较;如果相等,则找到了元素的位置。每次比较的数列长度都会是之前数列的一半,直到找到相等元素的位置或者最终没有找到要找的元素。

三、时间复杂度

我们先来想象一下,如果数列中有 3 个数,则先与第 2 个数进行比较,如果比第 2 个数大,则与第 2 个数右边的数列进行二分查找,这时这个数列就剩下一个数了,直接比较是否相等即可。所以在 3 个数的时候最多比较两次。

同理,在有 4 个数的时候,我们与中间数进行比较,一般中间数是首加末除以 2 算出来的,这时我们算出来的中间数是 (1+4)/2 等于 2,所以我们把要查找的数与第 2 个数比较,若比第 2 个数小,则直接与第 1 个数比较;否则与后面两个数进行二分查找,这时的中间数是 (3+4)/2 等于 3,也就是后半部分的第 1 个数。再接着进行比较,相等则找到相应的元素,小于则没有这个数(因为左边所有的数都已经判断过了),大于则继续向右查找。所以在 4 个数的时候最多比较 3 次。 以此类推,在 5 个数的时候最多查找 3 次,在 6 个数的时候也是最多查找 3 次。 时间复杂度:O(logN)

四、算法

画个图便于理解,我把一整个数组划分为三个部分,头部,尾部,中部。

在这里插入图片描述
在这里插入图片描述

我们每一次查找要么在后半区查找,要么在前半区查找。比如我要在前半区查找,那后半区我们就不要了,high就要移动到mid前一个,mid就要移动到low与mid之间。也就是形成一个新的减半数组。

4.1非递归思想

算法如下,具体再看注释:

代码语言:javascript
复制
//非递归思想
int search(SeqList r, int k)
{//在r[1....n]中查找k
    low = 1; high = n;
    while (low <= high)
    {
        mid = (low + high) / 2;
        if (k == r[mid].key)   return mid;//找到待查元素
        else if (k > r[mid].key)  low = mid + 1;//在后半区查找
        else high = mid - 1;//继续在前半区查找

    }
    return 0;//查找失败返回0
}

4.2递归思想

算法如下,具体再看注释:

代码语言:javascript
复制
//递归思想
in search1(SwqList r, int k, in low, int high)
{
    if (low > high) return 0;//排除不合法数组
    mid = (low + high) / 2;
    if (k == r[mid].key)  return mid;//找到待查找元素
    else if (k > r[mid].key)
        return search1(r, k, mid + 1, high);//low变成了原来的mid前,继续查找后半区
    else
        return search1(r, k, low, mid - 1);//继续查找前半区
}

五、Leecode案例

5.1例一

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

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

示例2:

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

我们套上面讲的算法模板,代码为:

代码语言:javascript
复制
int search(int* r, int n, int k)
{//在r[1....n]中查找k
   int low = 0;int high = n-1;
    while (low <= high)
    {
        int mid = (low + high) / 2;
        if (k == r[mid])   return mid;//找到待查元素
        else if (k > r[mid])  low = mid + 1;//在后半区查找
        else high = mid - 1;//继续在前半区查找

    }
    return -1;//查找失败返回0
}

执行结果:

在这里插入图片描述
在这里插入图片描述

5.2案例二

套算法模板!!! 题目:你是产品经理,目前正在带领一个团队开发新的产品。不幸的是,你的产品的最新版本没有通过质量检测。由于每个版本都是基于之前的版本开发的,所以错误的版本之后的所有版本都是错的。 假设你有 n 个版本 [1, 2, …, n],你想找出导致之后所有版本出错的第一个错误的版本。 你可以通过调用 bool isBadVersion(version) 接口来判断版本号 version 是否在单元测试中出错。实现一个函数来查找第一个错误的版本。你应该尽量减少对调用 API 的次数。 代码:

代码语言:javascript
复制
int firstBadVersion(int n) {
    int left = 1, right = n;
    while (left < right) {  // 循环直至区间左右端点相同
        int mid = left + (right - left) / 2;  // 防止计算时溢出
        if (isBadVersion(mid)) {
            right = mid;  // 答案在区间 [left, mid] 中
        } else {
            left = mid + 1;  // 答案在区间 [mid+1, right] 中
        }
    }
    // 此时有 left == right,区间缩为一个点,即为答案
    return left;
}

执行返回:

在这里插入图片描述
在这里插入图片描述

5.3案例三

题目:给定一个排序数组和一个目标值,在数组中找到目标值,并返回其索引。如果目标值不存在于数组中,返回它将会被按顺序插入的位置。请必须使用时间复杂度为 O(log n) 的算法。 还是套模板:

代码语言:javascript
复制
int searchInsert(int* nums, int numsSize, int target) {
    int left = 0, right = numsSize - 1, ans = numsSize;
    while (left <= right) {
        int mid = ((right - left) >> 1) + left;
        if (target <= nums[mid]) {
            ans = mid;
            right = mid - 1;
        } else {
            left = mid + 1;
        }
    }
    return ans;
}

返回:

在这里插入图片描述
在这里插入图片描述

六、 总结

二分查找就是一个静态的方法查表,折半比较,该方法的好处就是比较次数少,查找速度快,效率高。缺点就是必须要有序表,或者我们需要用排序方法先让它变成有序表。二分查找适用于不经常变动而查找频繁的有序表。 我是川川,如果能帮到你,那就在幸运不过了,一起加油!

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2021-08-24 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 一、算法详细讲解
    • 1.0 前言介绍
      • 1.1二分查找介绍
        • 1.2二分查找条件
        • 二、 原理及实现
        • 三、时间复杂度
        • 四、算法
          • 4.1非递归思想
            • 4.2递归思想
            • 五、Leecode案例
              • 5.1例一
                • 5.2案例二
                  • 5.3案例三
                  • 六、 总结
                  领券
                  问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档