news 2026/6/10 2:50:41

【算法题】二分

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
【算法题】二分

二分查找是高效解决有序/局部有序数组问题的经典算法,核心思想是通过不断缩小“可能包含目标的区间”,将时间复杂度从暴力遍历的O(n)O(n)O(n)优化到O(log⁡n)O(\log n)O(logn)
它的适用场景非常广泛:不仅能解决“查找目标值”这类基础问题,还能处理“找边界”“找极值”“旋转数组”等复杂场景。本文将通过7道经典题目,拆解二分查找在不同场景下的解题思路与代码实现。

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

题目描述:
给定非递减排序的整数数组nums和目标值target,找出target在数组中的开始位置和结束位置;若不存在则返回[-1, -1]。要求时间复杂度为O(log⁡n)O(\log n)O(logn)

示例

  • 输入:nums = [5,7,7,8,8,10], target = 8,输出:[3,4]
  • 输入:nums = [5,7,7,8,8,10], target = 6,输出:[-1,-1]

解题思路:
通过两次二分查找分别确定左边界和右边界:

  1. 找左边界:二分找第一个等于target的位置。若nums[mid] < target,左指针右移;否则右指针左移,最终左指针即为左边界。
  2. 找右边界:二分找最后一个等于target的位置。若nums[mid] <= target,左指针右移;否则右指针左移,最终右指针即为右边界。
  3. 若左边界对应的元素不是target,直接返回[-1,-1]

完整代码:

classSolution{public:vector<int>searchRange(vector<int>&nums,inttarget){if(nums.size()==0)return{-1,-1};intbegin=0;intleft=0,right=nums.size()-1;// 找左边界while(left<right){intmid=left+(right-left)/2;if(nums[mid]<target)left=mid+1;elseright=mid;}if(nums[left]!=target)return{-1,-1};elsebegin=left;// 找右边界right=nums.size()-1;while(left<right){intmid=left+(right-left+1)/2;if(nums[mid]<=target)left=mid;elseright=mid-1;}return{begin,right};}};

复杂度分析:

  • 时间复杂度:O(log⁡n)O(\log n)O(logn),两次二分查找各占O(log⁡n)O(\log n)O(logn)
  • 空间复杂度:O(1)O(1)O(1),仅用常数级额外变量。

二、x 的平方根

题目描述:
给定非负整数x,计算并返回其算术平方根的整数部分(舍去小数部分),不允许使用内置指数函数或运算符。

示例

  • 输入:x = 4,输出:2
  • 输入:x = 8,输出:2(8的平方根是2.828…,取整数部分)

解题思路:
通过二分查找找最大的整数mid,使得mid² <= x

  1. 边界条件:若x < 1,直接返回0;否则二分区间为[1, x]
  2. 二分过程:计算mid = left + (right - left + 1) / 2(避免死循环),用long long存储mid * mid防止整数溢出;若mid * mid <= x,左指针右移(保留当前候选值),否则右指针左移。

完整代码:

classSolution{public:intmySqrt(intx){if(x<1)return0;intleft=1,right=x;while(left<right){longlongmid=left+(right-left+1)/2;if(mid*mid<=x)left=mid;elseright=mid-1;}returnleft;}};

复杂度分析:

  • 时间复杂度:O(log⁡x)O(\log x)O(logx),二分区间大小为x,每次缩小一半。
  • 空间复杂度:O(1)O(1)O(1),仅用常数级额外变量。

三、搜索插入位置

题目描述:
给定排序数组nums和目标值target,找到target在数组中的索引;若不存在,则返回其按顺序插入的位置。要求时间复杂度为O(log⁡n)O(\log n)O(logn)

示例

  • 输入:nums = [1,3,5,6], target = 5,输出:2
  • 输入:nums = [1,3,5,6], target = 2,输出:1

解题思路:
通过二分查找找第一个大于等于target的位置

  • nums[mid] < target,说明target在右边,左指针右移;否则右指针左移。
  • 最终左指针即为目标位置(若nums[left] < target,则插入到left+1,否则插入到left)。

完整代码:

classSolution{public:intsearchInsert(vector<int>&nums,inttarget){intleft=0,right=nums.size()-1;while(left<right){intmid=left+(right-left)/2;if(nums[mid]<target)left=mid+1;elseright=mid;}returnnums[left]<target?left+1:left;}};

复杂度分析:

  • 时间复杂度:O(log⁡n)O(\log n)O(logn),二分遍历数组。
  • 空间复杂度:O(1)O(1)O(1),仅用常数级额外变量。

四、山脉数组的峰顶索引

题目描述:
给定山脉数组arr(先递增后递减),返回峰值元素的下标,要求时间复杂度为O(log⁡n)O(\log n)O(logn)

示例

  • 输入:arr = [0,1,0],输出:1
  • 输入:arr = [0,2,1,0],输出:1

解题思路:
山脉数组的峰值满足arr[mid] > arr[mid-1]arr[mid] > arr[mid+1],通过二分缩小范围:

  • 二分区间为[1, arr.size()-2](避免越界),比较arr[mid]arr[mid-1]
    • arr[mid] > arr[mid-1],说明峰值在右边,左指针右移。
    • 否则说明峰值在左边,右指针左移。
  • 最终左指针即为峰值下标。

完整代码:

classSolution{public:intpeakIndexInMountainArray(vector<int>&arr){intleft=1,right=arr.size()-2;while(left<right){intmid=left+(right-left+1)/2;if(arr[mid]>arr[mid-1])left=mid;elseright=mid-1;}returnleft;}};

复杂度分析:

  • 时间复杂度:O(log⁡n)O(\log n)O(logn),二分遍历数组。
  • 空间复杂度:O(1)O(1)O(1),仅用常数级额外变量。

五、寻找峰值

题目描述:
峰值元素是指严格大于左右相邻值的元素,给定数组nums,返回任意一个峰值的下标。假设nums[-1] = nums[n] = -∞,要求时间复杂度为O(log⁡n)O(\log n)O(logn)

示例

  • 输入:nums = [1,2,3,1],输出:2
  • 输入:nums = [1,2,1,3,5,6,4],输出:15

解题思路:
利用“边界为负无穷”的假设,通过二分找峰值:

  • 二分区间为[0, nums.size()-1],比较nums[mid]nums[mid+1]
    • nums[mid] > nums[mid+1],说明峰值在左边(包括mid),右指针左移。
    • 否则说明峰值在右边,左指针右移。
  • 最终左指针即为峰值下标(必然存在峰值)。

完整代码:

classSolution{public:intfindPeakElement(vector<int>&nums){intleft=0,right=nums.size()-1;while(left<right){intmid=left+(right-left)/2;if(nums[mid]>nums[mid+1])right=mid;elseleft=mid+1;}returnleft;}};

复杂度分析:

  • 时间复杂度:O(log⁡n)O(\log n)O(logn),二分遍历数组。
  • 空间复杂度:O(1)O(1)O(1),仅用常数级额外变量。

六、寻找旋转排序数组中的最小值

题目描述:
给定升序旋转后的数组nums(元素互不相同),返回数组中的最小元素,要求时间复杂度为O(log⁡n)O(\log n)O(logn)

示例

  • 输入:nums = [3,4,5,1,2],输出:1
  • 输入:nums = [4,5,6,7,0,1,2],输出:0

解题思路:
旋转后的数组分为“左升序段”和“右升序段”,最小值是右段的第一个元素:

  • 二分区间为[0, nums.size()-1],比较nums[mid]nums.back()(最后一个元素):
    • nums[mid] > nums.back(),说明mid在左段,最小值在右边,左指针右移。
    • 否则说明mid在右段,最小值在左边(包括mid),右指针左移。
  • 最终左指针即为最小值的下标。

完整代码:

classSolution{public:intfindMin(vector<int>&nums){intleft=0,right=nums.size()-1;while(left<right){intmid=left+(right-left)/2;if(nums[mid]>nums[nums.size()-1])left=mid+1;elseright=mid;}returnnums[left];}};

复杂度分析:

  • 时间复杂度:O(log⁡n)O(\log n)O(logn),二分遍历数组。
  • 空间复杂度:O(1)O(1)O(1),仅用常数级额外变量。

七、点名

题目描述:
班级n位同学的学号为0~n-1,点名结果记录于升序数组records,仅一位同学缺席,返回其学号。

示例

  • 输入:records = [0,1,2,3,5],输出:4
  • 输入:records = [0,1,2,3,4,5,6,8],输出:7

解题思路:
正常情况下records[mid] == mid,缺席的学号会打破该关系:

  • 二分区间为[0, records.size()-1],比较records[mid]mid
    • records[mid] == mid,说明缺席在右边,左指针右移。
    • 否则说明缺席在左边(包括mid),右指针左移。
  • 最终若records[left] == left,缺席学号为left+1;否则为left

完整代码:

classSolution{public:inttakeAttendance(vector<int>&records){intleft=0,right=records.size()-1;while(left<right){intmid=left+(right-left)/2;if(records[mid]==mid)left=mid+1;elseright=mid;}returnrecords[left]==left?left+1:left;}};

复杂度分析:

  • 时间复杂度:O(log⁡n)O(\log n)O(logn),二分遍历数组。
  • 空间复杂度:O(1)O(1)O(1),仅用常数级额外变量。
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/30 9:23:50

Burp Suite 插件 | SQL 注入自定义扫描和分析

工具介绍 SQL Injection Scout 是一个用于 Burp Suite 的扩展&#xff0c;专为帮助安全研究人员和开发人员检测和分析 SQL 注入漏洞而设计。该扩展提供了丰富的配置选项和直观的用户界面&#xff0c;便于用户自定义扫描和分析过程。 &#x1f4af; 功能特性 被动检测SQL&…

作者头像 李华
网站建设 2026/6/7 20:56:56

Linux系统编程——网络:从 OSI 到 UDP 通信实践

目录 一、OSI 与 TCP/IP 模型 1.OSI 模型&#xff08;7 层&#xff09; 2.TCP/IP 模型&#xff08;4 层&#xff09; 二、Linux 网络配置&#xff1a;命令与文件 1. 永久配置 IP 2.临时配置 IP 3.网络调试命令 三、网络编程核心概念&#xff1a;Socket 与字节序 1.Soc…

作者头像 李华
网站建设 2026/6/1 8:12:43

Excel超实用技能,一键批量快速合并相同内容单元格,瞬间搞定

经常使用Excel整理分析数据的小伙伴可能会有这样的烦恼,拿到一个工作簿后,工作表中有很多重复的数据,而且这些数据对应的数值可能各不相同,怎么快速合并这些相同内容的单元格呢? 熟知Excel操作的小伙伴,动动手指也就是几分钟的事情。使用高版本Excel的小伙伴更是几十秒就…

作者头像 李华
网站建设 2026/5/30 9:23:48

Java虚拟机面试题:内存管理(下)

&#x1f9d1; 博主简介&#xff1a;CSDN博客专家&#xff0c;历代文学网&#xff08;PC端可以访问&#xff1a;https://literature.sinhy.com/#/?__c1000&#xff0c;移动端可微信小程序搜索“历代文学”&#xff09;总架构师&#xff0c;15年工作经验&#xff0c;精通Java编…

作者头像 李华