【LeetCode 刷题笔记】35. 搜索插入位置 | 二分查找经典入门题


一、题目描述

给定一个排序数组和一个目标值,在数组中找到目标值,并返回其索引。如果目标值不存在于数组中,返回它将会被按顺序插入的位置。

请必须使用时间复杂度为 O(log n) 的算法。

示例 1:

复制代码
输入: nums = [1,3,5,6], target = 5
输出: 2

示例 2:

复制代码
输入: nums = [1,3,5,6], target = 2
输出: 1

示例 3:

复制代码
输入: nums = [1,3,5,6], target = 7
输出: 4

二、解题思路

  • 题目明确要求时间复杂度为 O(log n),因此暴力遍历(O(n))不符合要求,必须使用二分查找
  • 核心需求拆解:
    1. 若数组中存在目标值,直接返回其索引;
    2. 若目标值不存在,返回按升序插入的位置(本质是寻找数组中第一个大于目标值的元素的索引)。
  • 二分查找核心逻辑:通过不断缩小查找区间,每次排除一半元素,最终定位目标值或确定插入位置。

三、代码实现(Java)

解法1:

复制代码
class Solution {
    public int searchInsert(int[] nums, int target) {
        // 边界处理:数组为null的鲁棒性判断
        if(nums == null){
            return -1;
        }
        // 初始化二分查找区间(闭区间 [low, high])
        int low = 0;
        int high = nums.length - 1;
        int mid = 0;
        // 二分查找循环:low <= high 保证所有元素都被遍历
        while(low <= high){
            // 计算中间位置,避免 low + high 直接相加导致int溢出
            mid = low + (high - low) / 2;
            if(nums[mid] == target){
                // 找到目标值,直接返回索引
                return mid;
            }
            else if(nums[mid] < target){
                // 目标值更大,需向右半部分查找,更新左边界
                low = mid + 1;
            }
            else{
                // 目标值更小,需向左半部分查找,更新右边界
                high = mid - 1;
            }
        }
        // 退出循环时,low 即为目标值的插入位置
        return low;
    }
}

也可以用 【LeetCode 刷题笔记】34. 在排序数组中查找元素的第一个和最后一个位置 | 二分查找经典刷题题解这篇文章的思路,本质也还是二分查找。
解法2:代码如下:

复制代码
class Solution {
    public int mySqrt(int x) {
        int l = 0, r = x, ans = -1;
        while (l <= r) {
            int mid = l + (r - l) / 2;
            if ((long) mid * mid <= x) {
                ans = mid;
                l = mid + 1;
            } else {
                r = mid - 1;
            }
        }
        return ans;
    }
}

四、核心笔记&易错点解析

1. 边界情况处理:空数组/空输入

  • nums == null 时,返回-1(题目默认nums为有效排序数组,添加此判断可提升代码鲁棒性)。

  • nums.length == 0 时,high = nums.length - 1 = -1,此时循环条件 low <= high0 <= -1)不成立,直接退出循环,返回 low = 0,正好对应空数组的插入位置为0,符合题目要求。

2. 为什么循环退出后直接返回 low

很多同学会疑惑:循环结束时 low > high,此时low为什么一定是正确的插入位置?我们可以结合循环的最后一步(low == high时)分析:

low == high 时,mid = low = high,此时分三种情况:

  1. nums[mid] == target:已在循环内直接return mid,不会走到退出循环的步骤;

  2. nums[mid] < target :目标值比当前mid位置的元素大,插入位置应为 mid + 1。循环内会执行 low = mid + 1,此时 low > high,退出循环,low 正好等于 mid + 1,即正确插入位置;

  3. nums[mid] > target :目标值比当前mid位置的元素小,插入位置应为当前的mid(即low)。循环内会执行 high = mid - 1,此时 low > high,退出循环,low 仍为原来的mid值,即正确插入位置。

因此,无论哪种情况,循环退出后low一定是目标值的正确插入位置。

3. 二分查找细节优化

  • mid = low + (high - low) / 2:相比直接写 mid = (low + high) / 2,这种写法可避免 low + high 超过int最大值导致溢出,是二分查找的标准写法。

  • 循环条件 low <= high:这是闭区间二分查找的标准条件,保证数组所有元素都被遍历到,不会漏掉边界情况。


五、复杂度分析

  • 时间复杂度O(log n),每次循环将查找区间缩小一半,最多循环log₂n次,完全符合题目要求。

  • 空间复杂度O(1),仅使用了常数级别的额外空间。


六、总结

  • 这道题的核心是利用二分查找实现O(log n)的时间复杂度,关键在于理解循环退出后low的含义。

  • 闭区间二分查找的模板需要牢记,尤其是边界条件和mid的计算方式,避免溢出和死循环。

  • 这类"寻找插入位置"的问题,本质上是寻找数组中第一个大于等于目标值的元素的索引,用二分查找可以高效解决。

相关推荐
小欣加油1 小时前
leetcode56 合并区间
c++·算法·leetcode·职场和发展
lqqjuly1 小时前
前沿算法深度解析(二)
人工智能·算法·机器学习
徐小夕3 小时前
万字长文!千万级文档 RAG 知识库系统落地实践
前端·算法·github
akunkuntaimei3 小时前
2026年高考数学各省真题及答案(完整版)
算法·高考
Hello:CodeWorld4 小时前
C 风格变参 vs C++ 变参模板:核心区别与选型指南
c语言·c++·算法
8Qi85 小时前
LeetCode 516:最长回文子序列
算法·leetcode·职场和发展·动态规划
youngerwang6 小时前
【从搬运工到协处理器:网卡芯片架构、算法、验证与边缘演进深度剖析】
网络·算法·架构·芯片
KaMeidebaby6 小时前
卡梅德生物技术快报|纯化重组蛋白实操详解
人工智能·python·tcp/ip·算法·机器学习
手写码匠7 小时前
从零实现 Prompt 工程引擎:结构化提示、自动优化与多轮自省体系
人工智能·深度学习·算法·aigc
无限码力8 小时前
阿里算法岗 0530笔试真题 - 多约束条件下的元素匹配统计
算法·阿里笔试真题·阿里机试真题·阿里算法岗笔试