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

一、题目

给定一个非递减顺序排列的数组nums,和target,找出给定目标值在数组中的开始位置和结束位置,如果数组中不存在目标值target,返回[-1,-1]。时间复杂度O(log n)。

二、思路:

在有序数组中找重复元素的范围,就用两次二分:一次找左边界(≥ target),一次找右边界(≤ target)

利用两次二分查找 :第一次找第一个大于等于 target 的位置 (左边界),第二次找最后一个小于等于 target 的位置 (右边界);通过将 == 合并到 >=<= 条件中,确保即使找到目标值也继续向边界方向搜索,从而精准定位重复元素的范围,整体时间复杂度为 O(log n)。

三、代码

复制代码
class Solution {
    public int[] searchRange(int[] nums, int target) {
        int left = findLeft(nums, target);
        int right = findRight(nums, target);

        // 如果左边界 > 右边界,说明 target 不存在
        if (left > right) {
            return new int[]{-1, -1};
        }
        return new int[]{left, right};
    }

    // 找第一个 >= target 的位置(左边界)
    private int findLeft(int[] nums, int target) {
        int left = 0, right = nums.length - 1;
        int ans = nums.length; // 默认设为越界值(表示未找到)

        while (left <= right) {
            int mid = left + (right - left) / 2;
            if (nums[mid] >= target) {
                ans = mid;       // 记录可能的左边界
                right = mid - 1; // 继续向左找更小的
            } else {
                left = mid + 1;  // 太小了,去右边
            }
        }
        return ans;
    }

    // 找最后一个 <= target 的位置(右边界)
    private int findRight(int[] nums, int target) {
        int left = 0, right = nums.length - 1;
        int ans = -1; // 默认设为 -1(表示未找到)

        while (left <= right) {
            int mid = left + (right - left) / 2;
            if (nums[mid] <= target) {
                ans = mid;       // 记录可能的右边界
                left = mid + 1;  // 继续向右找更大的
            } else {
                right = mid - 1; // 太大了,去左边
            }
        }
        return ans;
    }
}

注:

if (nums[mid] >= target) 等于没有单独拿出来,是因为即使找到了 target,也不能停止,还要继续向左找更早的。

相关推荐
3秒一个大5 小时前
深入理解 JS 中的栈与堆:从内存模型到数据结构,再谈内存泄漏
前端·javascript·数据结构
xiaoyaohou115 小时前
003、轻量化改进(一):网络剪枝原理与实战
算法·机器学习·剪枝
我是章汕呐5 小时前
政策评估的“黄金标准”:DID模型从原理到Stata实操
大数据·人工智能·经验分享·算法·回归
2301_822703206 小时前
光影进度条:鸿蒙Flutter实现动态光影效果的进度条
算法·flutter·华为·信息可视化·开源·harmonyos
人道领域6 小时前
【LeetCode刷题日记】383 赎金信
算法·leetcode·职场和发展
炽烈小老头6 小时前
【每天学习一点算法 2026/04/11】Pow(x, n)
学习·算法
旖-旎6 小时前
哈希表(存在重复元素)(3)
数据结构·c++·学习·算法·leetcode·散列表
明月醉窗台6 小时前
[jetson] AGX Xavier 安装Ubuntu18.04及jetpack4.5
人工智能·算法·nvidia·cuda·jetson
计算机安禾6 小时前
【数据结构与算法】第39篇:图论(三):最小生成树——Prim算法与Kruskal算法
开发语言·数据结构·c++·算法·排序算法·图论·visual studio code
weixin_513449966 小时前
walk_these_ways项目学习记录第九篇(通过行为多样性 (MoB) 实现地形泛化)--学习算法
学习·算法·机器学习