【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的计算方式,避免溢出和死循环。

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

相关推荐
先吃饱再说11 小时前
判断回文字符串,从一行代码到双指针优化
算法
黄敬峰14 小时前
深入理解算法核心:从递归思想、数组扁平化到快速排序
算法
得物技术15 小时前
从狂野代码到按目标生产:得物推荐 AI Harness 的工程化实践|AICon 演讲整理
人工智能·算法·架构
AI小老六19 小时前
SkillOpt 架构拆解:把 Skill 文本当参数,用执行轨迹训练 Agent
后端·算法·ai编程
胡萝卜术19 小时前
从“分数打架”到“排名投票”:为什么你的ChatBI必须用RRF?
算法·设计模式·面试
Asize20 小时前
初识DFS 与 BFS:递归、队列与图遍历
算法
罗西的思考1 天前
机器人 / 强化学习】HIL-SERL:人类在环驱动的具身智能进化框架
人工智能·算法·机器学习
美团技术团队2 天前
LongCat 开源 VitaBench 2.0:长期动态智能体基准新标杆
人工智能·算法
To_OC2 天前
LC 207 课程表:刚学图论那会儿,我连这是拓扑排序都没看出来
javascript·算法·leetcode