【算法】手把手学会二分查找

news2024/9/28 3:27:42

目录

简介

基本步骤

第一种二分

第二种二分 

例题

搜索插入位置

数的范围

总结 


简介

🥥二分查找,又叫折半查找,通过找到数据二段性每次都能将原来的数据筛选掉一半,通过这个算法我们能够将一个一个查找的 O(n) 的时间复杂度优化到 O(logn) ,极大地提升了查找的效率。但使用二分进行查找必须要有一个前提,那就是查找的区间必须是有序的。如数组并非有序,则找不到该数组的的二段性。下面一起看看二分的基本步骤吧。

基本步骤

  • 找一个区间 [L , R],使答案一定在该区间中。
  • 找一个判断条件,使得该判断条件具有二段性,并且答案一定是该二段性的分界点。
  • 分析中点 mid 在该判断条件下是否成立,如果成立,考虑答案在哪个区间,如果不成立,答案在哪个区间
  • 如果更新方式为 R = mid, 则不做处理,若更新方式是L = mid,在计算 mid 时需要 +1

第一种二分

🥥假设当前我们有一个 1~9 的有序数组,现在我们要查找数组中的7。由此我们可以通过数字的大小将其分为小于 7 和 大于等于 7 的两个部分。

🥥若算出来 mid 在左区间,由于其左边的值都是小于 的所以不需要保留,便可以将迭代区间缩短至 [mid+1,R] 

🥥而如果 mid 在右区间,这个区间的范围是大于等于 的,当前的值是有可能等于 

🥥所以需要将当前 mid 位保留,所以递增区间便保留至[L , mid]。

🥥并且根据上面的基本步骤的最后一步,因为更新方式为 R = mid , 则计算mid的时候不做处理。因此 mid = (L+R) / 2 

int main()
{
	int arr[] = { 1,2,3,4,5,6,7,8,9 };
	int l = 0, r = 8,num = 7;
	while (l < r)
	{
		int mid = (l + r) / 2;  //迭代mid
		if (arr[mid] >= num)    //mid在右区间
		{
			r = mid;
		}
		else                    //mid在左区间
		{
			l = mid + 1;
		}
	}
	printf("%d\n", r);          //最后l和r一定相等
	return 0;
}

第二种二分 

🥥与上面那种不一样,这次我们将原数组分作小于等于7,以及大于7的两部分。

 🥥且这次若 mid 在左区间,由于该区间都是小于等于目标值的,因此该部分的数据需要保留,由此迭代区间至 [mid, R] 

🥥而当 mid 在右区间时,由于右区间并没有我们所需要的值,所以可以不用保留,所以迭代区间至 [L,mid-1]

🥥而现在,由于我们使用 L = mid 进行区间的更新,因此在计算 mid 的时候还需要加上1。

int main()
{
	int arr[] = { 1,2,3,4,5,6,7,8,9 };
	int l = 0, r = 8,num = 7;
	while (l < r)
	{
		int mid = (l + r + 1) / 2;   //计算mid时要+1
		if (arr[mid] <= num)         //mid在左区间
		{
			l = mid;                 //区间缩至[mid,r]
		}
		else                         //mid在右区间
		{
			r = mid - 1;             //区间缩至[l,mid-1]
		} 
	}
	printf("%d\n", r);               //最后l一定等于r
	return 0;
}

 🥥根据不同的二分法,二分查找有这两种不同的写法,因此在编写程序前,要先思考当前写法该如何迭代mid 在左区间时是怎样一种情况,在右区间又是什么情况。考虑好迭代关系,最后再处理 mid 的计算就相当简单,且不容易出错了。 

例题

搜索插入位置

传送门:搜索插入位置

🥥这题难度相对简单,要我们在数组中找目标值,若找不到则返回目标值若插入到这个数组时的所在的下标。

🥥用我们上面的思路进行分析,我们不妨将数组分作小于目标值的以及大于等于目标值两个区间。同时我们还要注意到目标值可能不存在或大于数组中的所有值,因此初始范围应当多扩展一位。由此便可得到代码。

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

数的范围

🥥传送门:AcWing 789. 数的范围

🥥通过读题我们可以知道,在这个数组之中有若干个重复的数,我们要根据题目找到一个数的区间,若无这个数则输出 -1 。但我们知道二分查找最后只能找一个数,那我们不妨这样想,因为这是个有序的数组,因此只要找到首尾两个端点就能够找到这个区间

🥥而首尾两个点就能够用二分查找找到。分别是将数组由小于目标值和大于等于目标值、小于等于目标值和大于目标值两种分法进行划分,即进行两次二分查找,而这两次二分查找的两个分界点恰好就是一个区间的两个端点

 🥥即经过两次二分,分别查找左边界及右边界(若只有一个数则左右边界相等),查找一个边界后我们还可以进行一次特判,若当前端点并非我们要求的目标值,则说明这个数组之中并没有我们想要的值,因此便可以直接输出 -1 ,否则就再继续查找另一端点。

#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>

const int N = 100000;
int n, m;
int arr[N];

int main()
{
	scanf("%d%d", &n, &m);
	for (int i = 0; i < n; i++)
	{
		scanf("%d", &arr[i]);
	}
	for (int i = 0; i < m; i++)  //  m组数据
	{
		int num;
		scanf("%d", &num);
		//求左端点
		int left = 0, right = n - 1;
		while (left < right)
		{
			int mid = (left + right) / 2;
			if (arr[mid] >= num)
			{
				right = mid;
			}
			else
			{
				left = mid + 1;
			}
		}
		if (arr[left] == num)
		{
			printf("%d ", left);
			//找右端点
			right = n - 1;
			while (left < right)
			{
				int mid = (left + right + 1) / 2;
				if (arr[mid] <= num)
				{
					left = mid;
				}
				else
				{
					right = mid - 1;
				}
			}
			printf("%d\n", left);
		}
		else
		{
			printf("-1 -1\n");
		}
	}
	return 0;
}

总结 

🥥二分查找的难点就在于边界的判断,因此每次在写代码前都要仔细思考,要如何进行二分?mid 在两个区间分别是什么情况?当前 mid 的值需不需要保留?最后在落实 mid 的计算。只有深刻地理解算法思想,在实际使用的时候才不会手忙脚乱。

🥥好了这次二分查找的入门讲解到这里就结束了,如果这篇文章对你有用的话还请留下你的三连加关注。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

从头搭建一个基于webpack的项目

从头搭建一个基于webpack的项目 一、起步 1、创建目录&#xff0c;初始化npm&#xff0c;安装webpack mkdir vue3-spa-templatecd vue3-spa-templatenpm init -ynpm install webpack webpack-cli --save-dev备注&#xff1a;在安装一个 package时&#xff0c;此 package 要…

一篇文章告诉你,为什么要使用Javascript流程图来可视化进程?(下)

DHTMLX Diagram库是有各种类型的图组成的&#xff0c;其中最广泛使用的是JavaScript流程图&#xff0c;它可以显示任何类型的的工作流、过程或系统&#xff0c;您可以下载DHTMLX Diagram的评估版并亲自试用。 在上文中&#xff08;点击这里回顾>>&#xff09;&#xff0…

Python标准库概览

Python标准库概览 知识点 标准库: turtle库(必选)标准库: random库(必选)、time库(可选&#xff09; 知识导图 1、turtle库概述 turtle&#xff08;海龟&#xff09;是Python重要的标准库之一&#xff0c;它能够进行基本的图形绘制。turtle库绘制图形有一个基本框架&#x…

ubuntu18.04复现yolo v8之最终章,realsenseD435i+yolo v8完美运行

背景&#xff1a;上一篇博客我们已经为复现yolov8配置好了环境&#xff0c;如果前面的工作顺利进行&#xff0c;我们已经完成了90%&#xff08;学习类程序最难的是环境配置&#xff09;。 接下来将正式下载yolov8的相关代码&#xff0c;以及进行realsenseD435i相机yolo v8的de…

ONLYOFFICE 文档如何与 Alfresco 进行集成

ONLYOFFICE 文档是一款开源办公套件&#xff0c;其是包含文本文档、电子表格、演示文稿、数字表单、PDF 查看器和转换工具的协作性编辑工具。要在 Alfresco 中使用 ONLYOFFICE 协作功能&#xff0c;可以将他们连接集成。阅读本文&#xff0c;了解这如何实现。 关于 ONLYOFFICE…

长胜证券:逾九成北交所公司上半年盈利,两大板块表现优异

到现在&#xff0c;已有28家北交所公司净赢利增速超越30%。 逾九成北交所公司完成盈余 半年报进入密布发表期。证券时报数据宝统计&#xff0c;到8月23日收盘&#xff0c;已有88家北交所上市公司发布2023年半年度相关陈述。 从营收规模来看&#xff0c;合计69家北交所公司上半…

传感器IMU

IMU根据其属性&#xff0c;可以知道其主要测量是线加速度和角加速度&#xff1a; (1) (2) 其中和分别是线加速度偏差和角加速度偏差,可以利用随机游走模型建模&#xff0c;, ,且符合高斯分布;和是噪声&#xff0c;一般认为符合高斯分布;表示全球坐标下的重力加速度;表示旋转变…

网络电子词典

一、项目要求&#xff1a; 1. 登录注册功能&#xff0c;不能重复登录&#xff0c;重复注册 2. 单词查询功能 3. 历史记录功能&#xff0c;存储单词&#xff0c;意思&#xff0c;以及查询时间 4. 基于TCP&#xff0c;支持多客户端连接 5. 采用数据库保存用户信息与历史记录…

002 编程是什么?

魔法师:在这个充满魔法和奇迹的数字时代,你是否好奇过计算机是如何运作的?当你用手机玩游戏、在电脑上浏览网页、看动画电影,你是否想过这背后的秘密是什么?别担心,今天我们将揭开这神秘的面纱,一起来探索编程的神奇世界! 编程,简单地说,就是一种让计算机执行任务的…

第五章:认证和动态菜单功能【基于Servlet+JSP的图书管理系统】

一、登录功能 1.认证实现 53-图书管理系统-登录功能-认证处理 首先完成最基础的登录功能&#xff0c;也就是在登录页面通过表单提交账号和密码到Servlet中。做相关的校验。给出不同的反应。 然后对应的Servlet中的处理逻辑 WebServlet(name "loginServlet",urlPatt…

软件测试的常用概念

目录 需求 需求和软件测试人员的关系 需求是侧式人员进行软件测试工作的依据,需要通过软件需求,来设计测试用例 软件的生命周期 在每个阶段,测试人员需要做什么事? 软件测试的生命周期 BUG 什么是bug? 如何描述一个bug? bug的级别 bug的生命周期: 调试和测试的区…

Linux虚拟机安装(Ubuntu 20)

最近这段时间使用VMWare安装了一下Ubuntu版本的Linux虚拟机&#xff0c;在这里记录一下安装时参考的文章以及需要注意的细节 参考链接&#xff1a; 虚拟机&#xff08;VMware&#xff09;安装Linux&#xff08;Ubuntu&#xff09;安装教程 VMware虚拟机下安装Ubuntu20.04&…

这所985重大科目变更!新增专硕可考信号!

一、学校及专业介绍 重庆大学&#xff08;ChongqingUniversity&#xff0c;CQU&#xff09;&#xff0c;简称“重大”&#xff0c;是教育部直属的全国重点大学&#xff0c;是国家“211工程”和“985工程”重点建设的高水平研究型综合性大学、国家“世界一流大学建设高校&#…

跨境电商系统开发:打破国界壁垒,拓展全球市场

拓展全球市场的必然选择 随着国际贸易水平的不断提升和全球市场的日益开放&#xff0c;跨境电商作为一种高效的贸易模式&#xff0c;受到了越来越多电商企业的关注和青睐。跨境电商不仅可以打破国界壁垒和时差限制&#xff0c;还可以在全球市场上实现销售的拓展&#xff0c;带来…

SVN 项目管理笔记

SVN 项目管理笔记 主要是介绍 SVN 管理项目的常用操作&#xff0c;方便以后查阅&#xff01;&#xff01;&#xff01; 一、本地项目提交到SVN流程 在SVN仓库下创建和项目名同样的文件夹目录&#xff1b;选中本地项目文件&#xff0c;选择SVN->checkout,第一个是远程仓库项…

FOC之SVPWM学习笔记

一、参考资料 【自制FOC驱动器】深入浅出讲解FOC算法与SVPWM技术 - 知乎FOC入门教程_zheng是在下的博客-CSDN博客DengFOC官方文档技术干货 |【自制】FOC驱动板 二、FOC控制算法流程框图 在FOC控制中主要用到三个PID环&#xff0c;从内到外依次是&#xff1a;电流环、速度环、位…

DevExpress WinForms数据编辑器组件,提供丰富的数据输入样式!(二)

DevExpress WinForms超过80个高影响力的WinForms编辑器和多用途控件&#xff0c;从屏蔽数据输入和内置数据验证到HTML格式化&#xff0c;DevExpress数据编辑库提供了无与伦比的数据编辑选项&#xff0c;包括用于独立数据编辑或用于容器控件(如Grid, TreeList和Ribbon)的单元格。…

第60步 深度学习图像识别:误判病例分析(Pytorch)

基于WIN10的64位系统演示 一、写在前面 上期内容基于Tensorflow环境做了误判病例分析&#xff08;传送门&#xff09;&#xff0c;考虑到不少模型在Tensorflow环境没有迁移学习的预训练模型&#xff0c;因此有必要在Pytorch环境也搞搞误判病例分析。 本期以SqueezeNet模型为…

Java不用加减乘除做加法(图文详解)

目录 1.题目描述 2.题解 分析 具体实现 1.题目描述 写一个函数&#xff0c;求两个整数之和&#xff0c;要求在函数体内不得使用、-、*、/四则运算符号。 示例 输入&#xff1a;1 2 输出&#xff1a;3 2.题解 分析 不能使用加减乘除四则运算符&#xff0c;那我们只能考虑…

论文笔记:从不平衡数据流中学习的综述: 分类、挑战、实证研究和可重复的实验框架

0 摘要 论文&#xff1a;A survey on learning from imbalanced data streams: taxonomy, challenges, empirical study, and reproducible experimental framework 发表&#xff1a;2023年发表在Machine Learning上。 源代码&#xff1a;https://github.com/canoalberto/imba…