【算法】二分查找——在排序数组中查找元素的第一个和最后一个位置

本节博客主要是通过“在排序数组中查找元素的第一个和最后一个位置”总结关于二分算法的左右界代码模板,有需要借鉴即可。

目录

  • 1.题目
  • 2.二分边界算法
    • 2.1查找区间左端点
      • 2.1.1循环条件
      • 2.1.2求中点的操作
      • 2.1.3总结
    • 2.2查找区间右端点
      • 2.1.1循环条件
      • 2.1.2求中点的操作
      • 2.1.3总结
    • 2.3总结
  • 3.参考解题代码
  • 4.模板总结
  • 5.总结

1.题目

题目链接:LINK
在这里插入图片描述

这个题要求我们求这个排序数组的一个元素的开始位置与结束位置。

可以用暴力求解的方法,把第一次出现的数字下标记录一下,最后一次记录一下,返回结果,除了复杂度差之外没什么不好的。

当然我们这里说一下二分算法的思想。之所以可以使用二分算法,这是因为该数组是有序的,可以利用二分算法的“二段性”将其分割。


用两次二分算法:

  • 一方面,我们可以将整个数组分为大于等于t和小于t来找left点

  • 另一方面,我们可以将整个数组分为大于t和小于等于t来找right点

但是这里有一些代码细节值得注意!!!

2.二分边界算法

2.1查找区间左端点

在这里插入图片描述

思考:我们在寻找左端点时候为什么要对数组按照小于t和大于等于t进行划分?
答:关键是因为我们要找左端点,左端点一定不可能在小于t的区间里。
在这里插入图片描述

通过上面的图片可知,我们要想找到一个数的左端点,那么这个左端点(我们要寻找的点)一定不再大于t这个区域,所以我们可知

  • mid < ret时,left = mid + 1
  • mid >= ret时,right = mid

2.1.1循环条件

while(left < right)//... √
while(left <= right)//... ×

循环条件选:left < right

这里为什么不是left <= right 呢?

  • left==right的情况下,即是最后结果,无需进行重复判断。
  • 可能有些情况下会出现死循环问题
    下面是对上面两个理由进行论证:
    在所有可能情况中,无非存在三种情况,
  • ①left与right中间存在要找的ret点
    在这里插入图片描述
    此时,mid = ret,mid == right,那么left = mid,会不断进入循环,陷入死循环
  • ②left与right中间所有点全部大于我们要找的右端点
    在这里插入图片描述
    到了最后,mid > ret, mid = right,right = mid,会存在死循环问题
  • ③left与right中间所有点全部小于我们要找的右端点
    在这里插入图片描述
    mid < ret,left = mid + 1,不会出现死循环问题。

2.1.2求中点的操作

我们求中点无非两种求法

①mid = left + (right - left) / 2; √
②mid = left + (right - left + 1) / 2; ×

这俩主要区别就是在数字个数是偶数情况下,①式取靠左的中点;②式取靠右的中点。

然后对于查找区间右端点而言,必须选用①式。
为什么,下面来进行解释?
如果选用②式,会存在下面情况:比如,mid指向right,然后mid所在的值>=ret值,就会不断死循环
注:if mid >= ret,right = mid;

在这里插入图片描述

2.1.3总结

在求目标值左端点时候,第一循环条件不能有等于,第二是求中点要用靠右中点。

2.2查找区间右端点

在这里插入图片描述

通过上面的图片可知,我们要想找到一个数的左端点,那么这个右端点(我们要寻找的点)一定不再大于t这个区域,所以我们可知

  • mid <= ret时,left = mid
  • mid > ret时,right = mid - 1

2.1.1循环条件

while(left < right)//... ×
while(left <= right)//... √

循环条件选:left < right

这里为什么不是left <= right 呢?

  • left==right的情况下,即是最后结果,无需进行重复判断。
  • 可能有些情况下会出现死循环问题

下面是对上面两个理由进行论证:
在所有可能情况中,无非存在三种情况,

  • ①left与right中间存在要找的ret点
    在这里插入图片描述
    此时,mid = ret,mid == right,那么right = mid,会不断进入循环,陷入死循环。
  • ②left与right中间所有点全部大于我们要找的右端点
    在这里插入图片描述
    到了最后,mid > ret,mid == right,right = mid - 1,不会出现死循环问题
  • ③left与right中间所有点全部小于我们要找的右端点
    在这里插入图片描述
    mid < ret,left = mid,此时会出现死循环问题

2.1.2求中点的操作

我们求中点无非两种求法

①mid = left + (right - left) / 2; ×
②mid = left + (right - left + 1) / 2;

这俩主要区别就是在数字个数是偶数情况下,①式取靠左的中点;②式取靠右的中点。

然后对于查找区间右端点而言,必须选用②式。
为什么,下面来进行解释?
如果选用①式,会存在下面情况:比如,mid指向left,然后mid所在的值<=ret值,left = mid,如此就会不断死循环
注:if mid <= ret,left = mid;
在这里插入图片描述

2.1.3总结

在求目标值右端点时候,第一循环条件不能有等于,第二是求中点要用靠右中点。

2.3总结

找左端点:
在这里插入图片描述

  • mid < ret时,left = mid + 1
  • mid >= ret时,right = mid
while(left < right)//...
mid = left + (right - left) / 2;

找右端点:
在这里插入图片描述

  • mid <= ret时,left = mid
  • mid > ret时,right = mid - 1
while(left < right)//...
mid = left + (right - left + 1) / 2;

根据上面的算法总结我们可以解决上面题目

3.参考解题代码

class Solution {
public:vector<int> searchRange(vector<int>& nums, int target) {vector<int> ret;//处理特殊情况if(nums.size() == 0){ret.push_back(-1);ret.push_back(-1);return ret;}int left = 0, right = nums.size() - 1;//处理左端点while(left < right){int mid = left + (right - left) / 2;if(nums[mid] >= target){right = mid;}else{left = mid + 1;}}if(nums[left] == nums[right] && nums[left] == target){ret.push_back(left);}else{ret.push_back(-1);}//处理右端点left = 0, right = nums.size() - 1;while(left < right){int mid = left + (right - left + 1) / 2;if(nums[mid] > target){right = mid - 1;}else{left = mid;}}if(nums[left] == nums[right] && nums[right] == target){ret.push_back(right);}else{ret.push_back(-1);}return ret;}
};

4.模板总结

在这里插入图片描述

5.总结

这个题目我感觉掌握了二分边界代码原理其实不难,重点肯定是那个二分边界算法原理,需要自己多理解一下。


EOF

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

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

相关文章

O2OA平台流程催办怎么做

O2OA平台设计了灵活的消息提醒数据交互方式&#xff0c;开发者可以根据自己的需要&#xff0c;来消费消息提醒数据&#xff0c;也可以将消息提醒数据接入到Kafka消息中间件来实现消息的准实时提醒。本篇主要介绍如何在O2OA服务器中设置流程的催办提醒消息。 催办提醒服务&#…

centos无法联网解决方案(9步完成

1.打开终端&#xff0c;输入 su - root 进入到管理员模式&#xff08;-的前后都有空格哈&#xff09; 切换后&#xff0c;显示的就是root... 2.. &#xff0c;输入命令ip addr 2. 切换当前目录 cd /etc/sysconfig/network-scripts/ 3.输入命令&#xff0c;打开文件 vi /etc…

一.常见算法--动态规划

&#xff08;1&#xff09;0-1背包问题 问题描述&#xff1a; 0-1背包问题的描述&#xff1a;在n种物品中选择1个或0个第i种物品&#xff0c;装入背包容量为m的背包&#xff0c;使得背包价值达到最大。 思路与关键点&#xff1a; 用到了max函数&#xff0c;用于返回两个数之中…

为何Linux成为你不可或缺的技能

在数字化飞速发展的今天&#xff0c;无论你是IT行业的精英&#xff0c;还是其他领域的专业人士&#xff0c;掌握Linux都已经成为一项至关重要的技能。那么&#xff0c;为什么一定要学会Linux呢&#xff1f;以下文章仅供参考 1. 开源的力量&#xff1a;无限的可能性 Linux是一…

工厂自动化升级改造(3)-Modbus与MQTT的转换

什么是MQTT,Modbus,见下面文章 工厂自动化升级改造参考(01)--设备通信协议详解及选型-CSDN博客文章浏览阅读608次,点赞9次,收藏6次。>>特点:基于标准的以太网技术,使用TCP/IP协议栈,支持高速数据传输和局域网内的设备通信。>>>特点:跨平台的通信协议,…

java版数字藏品深色UI仿鲸探数藏盲盒合成短视频卡牌模式支持高并发功能介绍

根据您提供的艺术品发售系统的需求&#xff0c;以下是一个更为详细和全面的系统设计概述&#xff1a; 1. 藏品发售 藏品分类&#xff1a;藏品可以按照不同的类别进行分类&#xff0c;如绘画、雕塑、摄影等。稀有度设置&#xff1a;后台可以为每个藏品设置不同的稀有度&#x…

ssl证书价格一年多少钱?如何申请?

由于行业新规&#xff0c;现在阿里云、腾讯云等几乎所有平台都不再提供一年期免费证书&#xff0c;如果需要一年期证书则需要支付一定的费用。SSL证书的价格根据类型不同几十到几百上千不等。 一年期SSL证书申请通道https://www.joyssl.com/?nid16 一年期SSL证书申请流程&am…

人工智能(一)架构

一、引言 人工智能这个词不是很新鲜&#xff0c;早就有开始研究的&#xff0c;各种推荐系统、智能客服都是有一定的智能服务的&#xff0c;但是一直都没有体现出多高的智能性&#xff0c;很多时候更像是‘人工智障’。 但是自从chatGpt3被大范围的营销和使用之后&#xff0c;人…

基于springboot的中小型医院网站源码数据库

基于springboot的中小型医院网站源码数据库 本基于Spring Boot的中小型医院网站设计目标是实现用户网络预约挂号的功能&#xff0c;同时提高医院管理效率&#xff0c;更好的为广大用户服务。 本文重点阐述了中小型医院网站的开发过程&#xff0c;以实际运用为开发背景&#x…

基于PHP+MySQL开发的百娣美业课程管理软件系统后端功能介绍

如何开发一个美容产业链的商户管理系统。 1. 需求分析。 在开发美容产业链商户管理系统之前&#xff0c;必须首先进行需求分析。商户需要明确自己的需求和目标&#xff0c;了解系统的功能模块和业务流程&#xff0c;为后续发展提供明确的方向。 2. 系统设计。 根据需求…

python常用基础知识

目录 &#xff08;1&#xff09;print函数 &#xff08;2&#xff09;注释 &#xff08;3&#xff09;input函数 &#xff08;4&#xff09;同时赋值和连续赋值 &#xff08;5&#xff09;type函数和id函数 &#xff08;6&#xff09;python赋值是地址赋值 &#xff08;…

Qt编译和使用freetype矢量字库方法

在之前讲过QT中利用freetype提取字库生成图片的方法&#xff1a; #QT利用freetype提取字库图片_qt freetype-CSDN博客文章浏览阅读1.2k次。这是某个项目中要用到的片段&#xff0c;结合上一篇文章#QT从字体名获取字库文件路径使用// 保存位图int SaveBitmapToFile(HBITMAP hBi…

【会议征稿,ACM出版】第四届人工智能,大数据与算法国际学术会议 (CAIBDA 2024, 7/5-7)

由河南省科学院、河南大学主办&#xff0c;河南省科学院智慧创制研究所、河南大学学术发展部、河南大学人工智能学院承办的第四届人工智能&#xff0c;大数据与算法国际学术会议 (CAIBDA 2024)将于2024年7月5-7日于中国郑州隆重举行。CAIBDA 2024致力于为人工智能&#xff0c;大…

你还去营业厅注销流量卡吗?别浪费时间了,现在有三种方法都可以

家人们&#xff0c;不用的手机号你会注销吗&#xff1f;在这里小编提醒大家&#xff0c;不用的手机卡千万要记得注销&#xff0c;不能直接扔掉&#xff0c;不然可能收到天价欠费单。 ​  今天&#xff0c;小编整理汇总了三种常见的注销方法&#xff0c;不用再跑去营业厅了&a…

Spring的IOC(Inversion of Control)设计模式

Spring的IOC&#xff08;Inversion of Control&#xff09;是一种设计模式&#xff0c;它通过控制反转的思想来降低组件之间的耦合度。在Spring框架中&#xff0c;IOC容器负责管理应用程序中的对象&#xff0c;使得对象之间的依赖关系由容器来维护和注入。 以下是Spring IOC的…

google hack常用命令举例

Google 常用语法说明 site 指定域名 inurl URL 中存在的关键字页面 intext 网页内容里面的关键字 Filetype 指定文件类型 intitle 网页标题中的关键字 link 返回你所有的指定域名链接 info 查找指定站点信息 cache 搜索 Google 里的内容缓存 技巧 1&#xff1a;inurl&…

MySQL入门学习-数据查询.SELECT

MySQL是一种常用的关系型数据库管理系统&#xff0c;可以用于存储和管理大量的结构化数据。在MySQL中&#xff0c;SELECT语句用于从数据库中查询数据。它是最常用的SQL语句之一&#xff0c;具有以下特点&#xff1a; 1. SELECT语句用于从一个或多个表中选择数据&#xff0c;并…

拓展(华为优秀网站)

优秀网站示例&#xff1a; 鸿蒙内核源码分析 Feature Ability和AbilitySlice的关系 使用绘图组件Canvas绘制心率曲线图 多组示例演示三个样式的组合用法 列表组件ListContainer Linux下的Hi3861一站式鸿蒙开发烧录&#xff08;附工具&#xff09; 全球首发—鸿蒙开源平台…

碳纳米管须状触嗅觉多模态融合传感器在皮革奢侈品真伪鉴定下的设计探索

一、设计方案 1.传感器选择 触觉传感器&#xff1a;选择基于碳纳米管&#xff08;CNT&#xff09;聚合物的柔性MEMS触觉微传感器&#xff0c;由于碳纳米管具有高度的灵敏度和选择性、柔韧性&#xff0c;可以作为触觉传感器&#xff0c;检测材料的微观结构和机械特性。嗅觉传感…

网页打开:为什么国内用新标签页,国外用当前页?

想写这个话题很久了&#xff0c;因为用百度和Google搜索时&#xff0c;打开搜索结果链接时的交互差异&#xff0c;几乎每天都要提醍我一下。 网页打开——这个交互&#xff0c;在设计里&#xff0c;算是极微小&#xff0c;但影响极广泛的操作设计。甚至&#xff0c;因此形成了…