news 2026/2/5 0:43:29

二分查找的应用

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
二分查找的应用

二分查找的应用

在学习本帖子之前,请确保你以及学习了上一篇二分查找,明白了二分查找的过程和基本写法,以及循环不变量的含义

现在我们再看看力扣第34题,关于二分查找的应用。

思路

题目是这样的

给你一个按照非递减顺序排列的整数数组 nums,和一个目标值 target。请你找出给定目标值在数组中的开始位置和结束位置。
如果数组中不存在目标值 target,返回 [-1, -1]。
你必须设计并实现时间复杂度为 O(log n) 的算法解决此问题。

很显然,只要遍历整个数组,我们就能很轻松的得到答案,但是遍历的复杂度是O(n),不符合题目的要求。

这道题也是在一个有序的数组中查找一个元素,但是问题是,他不是返回元素出现的位置即可,他要求找到元素在数组中第一次出现的位置和最后一次出现的位置。我们同样可以使用二分查找的思路来解决这个问题。

重新定义left和right

说是使用了二分查找,其实只是框架和二分查找一样,但是left和right的语义却不同。或者我们换一个思路来看一看之前的程序,看一看,如果target不在数组中,程序结束时,left和right的值是什么,以及为什么left和right的值会有这样的性质。

先说结论,如果target的值不在数组中,left(或者right,或者left和right)的值就会指向一个可以让值为target的数插入,而且不会破坏数组有序性的位置。

这里所谓插入,是指将第i个位置的值变为target,然后原来的第i个数往后移动的情况。
例如一个数组[1,5,7,7,8,8,10],在第4个位置(从0开始计数)插入一个6,变为[1,5,7,7,6,8,8,10]

我们来一个难的,也就是[left,right]这个区间的结束情况。
为什么说这个难呢,因为这算法如果跳出了循环,那么left和right一定不一样的,这样就要对他们两个都进行解释。如果是[left,right),循环结束,left一定等于right,解释一个即可。

intbinary_search(vector<int>&nums,inttarget){intleft=0,right=nums.size()-1;while(left<=right){intmid=(left+right)/2;if(target==nums[mid]){// 假设不会来到这里,因为target不在数组中assert(false)returnmid;}elseif(target<nums[mid]){// 将区间变为[left,mid-1]// 因为nums[mid]已经确定了不是targetright=mid-1;}elseif(target>nums[mid]){// 将区间变为[mid+1,right]// 因为nums[mid]已经确定了不是targetleft=mid+1;}}// 没有找到return-1;}

现在我要求你不要再关注区间内的元素,看看区间外的元素。

1 <-left 2 3 7 7 10 <- right

假设我们要找的数是12,其实很显然,那么指针一定是这样结束的

1 2 3 7 7 10 <- right <- left

那么这时left的值是什么含义呢?left指针指向了第一个12可以插入的位置。

为什么会这样呢?我们换个角度看看left,将它的语义定义为[0,left)指针的值都是小于target的值。我们是怎么维护它的语义的呢?

elseif(target>nums[mid]){// 在得知target大于nums[mid]后,我们将left移动到右边// 也就是说left左边的值都是小于target的left=mid+1;}

right恰好相反。在区间(right,nums.size()]的所有元素都是大于target的。这同样可以从right的更新过程中看出来。

elseif(target<nums[mid]){// 如果target确定小于nums[mid]// 也就是说right右边的值都是大于target的right=mid-1;}

但是等于target的值都在哪里呢?由于这个算法在遇到target之后直接返回了,所以没有定义等于target的值应该出现在哪个区间中。
于是我们可以在target==nums[mid]时不返回,而是修改left或者right,让等于target的值也出现在特定的区间中,达到特的那个目的。

假如我要使left左边的值都小于等于target,该怎么做呢?
很简单,只要

if(target==nums[mid]){// 确定了nums[mid]的值等于target// 那么就让他出现在left的左边即可left=mid+1;}

做了这么一个简单的修改,你就能保证,在任意一次循环中,我们都能确定left左边的值都是小于等于target的值,right右边的值都是大于target的值。在循环结束后,left就指向一个target可以插入的最后一个位置。

例如搜索能插入8的最后一个位置,则循环结束时,left和right如下所示

1 5 7 7 8 8 <- right 10 <- left

相信聪明的你也能知道这么修改代码使得left左边的值都是小于target,right右边的值都是大于等于target的值了,这样通过left你就能找到第一个target可以插入的值,如下所示

1 5 7 7 <- right 8 <- left 8 10

只要精准地控制left和right的语义,你就能得到你想要的值

同时,你也可以思考,为什么修改left和right的语义之后,循环条件为什么不变呢?

提示,你可以从确定了的元素个数考虑,如果所有的元素都已经确定的区间,那么就可以停止搜索

总结

从二分查找中,我们不止可以学习二分查找,还可以了解一个重要的概念——循环不变量。是否正确维护不变量,对于程序的正确运行至关重要。学会了这个,剩下的半开半闭区间的搜索模式,你就可以自由探索了。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/2/4 12:30:56

基于java+ vue农产投入线上管理系统(源码+数据库+文档)

农产投入线上管理 目录 基于springboot vue农产投入线上管理系统 一、前言 二、系统功能演示 三、技术选型 四、其他项目参考 五、代码参考 六、测试参考 七、最新计算机毕设选题推荐 八、源码获取&#xff1a; 基于springboot vue农产投入线上管理系统 一、前言 博…

作者头像 李华
网站建设 2026/2/1 19:54:32

LangFlow能否用于构建智能客服质检系统?会话分析实践

LangFlow在智能客服质检系统中的实践&#xff1a;从会话分析到高效落地 在现代客户服务场景中&#xff0c;每天产生的对话数据量呈指数级增长。无论是电商平台的售前咨询、金融行业的售后服务&#xff0c;还是电信企业的投诉处理&#xff0c;企业都面临着一个共同难题&#xff…

作者头像 李华
网站建设 2026/2/4 18:46:49

Open-AutoGLM加载慢怎么办?3种高效优化策略立即见效

第一章&#xff1a;Open-AutoGLM页面加载缓慢的根源剖析Open-AutoGLM作为一款基于AutoGLM架构的开源自动化工具平台&#xff0c;其页面加载性能直接影响用户体验。在实际部署中&#xff0c;部分用户反馈页面响应延迟显著&#xff0c;尤其在高并发或弱网络环境下表现尤为明显。该…

作者头像 李华
网站建设 2026/2/4 14:23:51

LangFlow能否用于构建AI心理咨询师原型?伦理边界探讨

LangFlow能否用于构建AI心理咨询师原型&#xff1f;伦理边界探讨 在一场深夜的线上心理援助测试中&#xff0c;一位志愿者输入了“我今天又想自我伤害了”——系统没有立刻生成安慰语句&#xff0c;而是先触发了一个红色标记的危机干预节点&#xff0c;随后返回一条温和但坚定的…

作者头像 李华
网站建设 2026/2/3 2:21:12

Open-AutoGLM权限管理深度优化(90%开发者忽略的弹窗隐患)

第一章&#xff1a;Open-AutoGLM权限管理深度优化&#xff08;90%开发者忽略的弹窗隐患&#xff09;在集成 Open-AutoGLM 框架时&#xff0c;多数开发者仅关注功能实现&#xff0c;却忽视了权限请求时机与用户感知之间的平衡。不当的权限弹窗策略不仅会降低用户体验&#xff0c…

作者头像 李华
网站建设 2026/2/4 8:05:29

权限弹窗频发怎么办,一文掌握Open-AutoGLM无感授权处理方案

第一章&#xff1a;权限弹窗频发的根源与挑战在现代移动和桌面应用开发中&#xff0c;权限弹窗频繁出现已成为影响用户体验的重要问题。系统为保护用户隐私&#xff0c;默认对敏感资源&#xff08;如位置、相机、通讯录&#xff09;进行访问限制&#xff0c;应用每次请求这些资…

作者头像 李华