当前位置: 首页 > news >正文

【LeetCode 题解】算法:34.在排序数组中查找元素的第一个和最后一个位置

一、问题剖析

题目要求

给定一个非递减数组 nums 和一个目标值 target,需要找到:

  • 目标值在数组中的起始位置

  • 目标值在数组中的结束位置
    如果目标值不存在,返回 [-1, -1]

关键约束

  • 时间复杂度必须为 O (log n):排除线性扫描的可能

  • 数组可能包含重复元素:需要处理多个相同元素的情况

  • 数组可能为空或全为负数:需考虑边界条件

二、核心思路:两次二分查找

1. 常规二分查找的局限性

普通二分查找只能找到一个匹配的位置,但无法确定是否为第一个或最后一个出现的位置。例如,数组 [5,7,7,8,8,10] 中查找 8,常规二分可能返回索引 3 或 4。

2. 左右边界二分法

通过两次独立的二分查找分别确定:

  • 左边界:第一个等于 target 的位置

  • 右边界:最后一个等于 target 的位置

左边界查找逻辑
初始化 left=0, right=n-1
当 left <= right 时:
    mid = (left + right) / 2
    if nums[mid] >= target:
        right = mid - 1
    else:
        left = mid + 1
最终检查 nums[left] 是否等于 target
右边界查找逻辑
初始化 left=0, right=n-1
当 left <= right 时:
    mid = (left + right) / 2
    if nums[mid] <= target:
        left = mid + 1
    else:
        right = mid - 1
最终检查 nums[right] 是否等于 target

三、Java 代码实现

​
class Solution {
    public int[] searchRange(int[] nums, int target) {
        int left = findLeftBound(nums, target);
        int right = findRightBound(nums, target);
        
        if (left <= right && nums[left] == target) {
            return new int[]{left, right};
        }
        return new int[]{-1, -1};
    }

    private int findLeftBound(int[] nums, int target) {
        int left = 0, right = nums.length - 1;
        while (left <= right) {
            int mid = left + (right - left) / 2;
            if (nums[mid] >= target) {
                right = mid - 1;
            } else {
                left = mid + 1;
            }
        }
        return left;
    }

    private int findRightBound(int[] nums, int target) {
        int left = 0, right = nums.length - 1;
        while (left <= right) {
            int mid = left + (right - left) / 2;
            if (nums[mid] <= target) {
                left = mid + 1;
            } else {
                right = mid - 1;
            }
        }
        return right;
    }
}

​

四、代码详细解析

1. 主方法 searchRange

  • 调用两次二分查找:分别获取左边界和右边界

  • 有效性验证:检查左边界是否在有效范围内且对应值等于目标值

2. 左边界查找 findLeftBound

  • 调整条件:当 nums[mid] >= target 时,说明左边界可能在左侧

  • 循环结束后left 指向第一个大于等于 target 的位置

3. 右边界查找 findRightBound

  • 调整条件:当 nums[mid] <= target 时,说明右边界可能在右侧

  • 循环结束后right 指向最后一个小于等于 target 的位置

五、复杂度分析

时间复杂度

  • 两次二分查找:每次时间复杂度为 O (log n)

  • 总时间复杂度:O(log n)

空间复杂度

  • 常数额外空间:O(1)

六、测试用例验证

测试用例 1

输入nums = [5,7,7,8,8,10], target = 8

  • 左边界查找:最终 left=3

  • 右边界查找:最终 right=4

  • 输出[3,4]

测试用例 2

输入nums = [5,7,7,8,8,10], target = 6

  • 左边界查找:最终 left=0(nums[0]=5 <6)

  • 右边界查找:最终 right=-1

  • 输出[-1,-1]

测试用例 3

输入nums = [], target = 0

  • 左边界查找:返回 0

  • 右边界查找:返回 -1

  • 验证0 > -1,返回 [-1,-1]

七、常见问题与优化

问题 1:为什么需要两次二分查找?

  • 左边界:在找到第一个匹配点后,需要继续向左查找

  • 右边界:在找到最后一个匹配点后,需要继续向右查找

问题 2:如何处理数组为空?

  • 直接返回 [-1,-1]

优化点:合并两次查找

可以将两次查找合并为一次,但会降低代码可读性。建议保持两次独立查找,逻辑更清晰。

感谢各位的阅读,后续将持续给大家讲解力扣中的算法题和数据库题,如果觉得这篇内容对你有帮助,别忘了点赞和关注,后续还有更多精彩的算法解析与你分享!

http://www.dtcms.com/a/118553.html

相关文章:

  • Kafka 中的 offset 提交问题
  • Qt 资源文件(.qrc 文件)
  • 基于SpringBoot的“高校社团管理系统”的设计与实现(源码+数据库+文档+PPT)
  • 基于ensp的mpls的解决bgp域内黑洞及MPLS VPN的应用
  • 心脏滴血漏洞(CVE-2014-0160)漏洞复现
  • 探秘PythonJSON解析深度剖析json.loads处理嵌套JSON字符串的奥秘
  • 《UNIX网络编程卷1:套接字联网API》第3章 套接字编程简介
  • MBR的 扩展分区 和 逻辑分区 笔记250407
  • 循环神经网络 - 机器学习任务之同步的序列到序列模式
  • 计算机网络学习前言
  • 八、C++速通秘籍—动态多态(运行期)
  • 【蓝桥杯】搜索算法:剪枝技巧+记忆化搜索
  • SpringBoot类跨包扫描失效的几种解决方法
  • SpringBoot企业级开发之【用户模块-登录】
  • 群晖NAS的最好的下载软件/电影的方式(虚拟机安装win系统安装下载软件)
  • 【5分钟论文阅读】InstructRestore论文解读
  • linux-core分析 : sip变量赋值-指针悬挂
  • 【LeetCode】算法详解#3 ---最大子数组和
  • 人工智能新时代:从深度学习到自主智能
  • 人工智能:深度学习关键技术与原理详解
  • LeetCode 解题思路 30(Hot 100)
  • 硬盘分区格式之GPT(GUID Partition Table)笔记250407
  • 【k8s学习之CSI】理解 LVM 存储概念和相关操作
  • 喂饭教程-Dify如何集成RAGFlow知识库
  • [ISP] ISP 中的 GTM 与 LTM:原理、算法与与 Gamma 校正的对比详解
  • Token+JWT+Redis 实现鉴权机制
  • 2024年十五届蓝桥杯青少年Scratch省赛初级组——找不同
  • 极空间NAS进阶玩法:Debian 系统安装教程
  • Docker学习--卷相关命令
  • 瓦片数据合并方法