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

在排序数组中查找元素的第一个和最后一个位置(力扣)

一.题目介绍

二.题目解析

使用二分进行查找

2.1处理边界情况

如果数组为空,直接返回 [-1, -1],因为无法找到目标值。

int[] ret = new int[2];
ret[0] = ret[1] = -1;
if (nums.length == 0) return ret;

2.2查找左端点(目标值开始位置)

1.初始化指针:

  • left 指向数组的起始位置(索引 0)。
  • right 指向数组的结束位置(索引 nums.length - 1)。

2.二分查找:

  • 计算中间位置 mid = left + (right - left) / 2 (细节一,不然可能陷入死循环)
  • 如果 nums[mid] < target,说明目标值在右半部分,更新 left = mid + 1。
  • 如果 nums[mid] >= target,说明目标值在左半部分或当前位置就是目标值,更新 right = mid。

 

3.循环退出并判断:

  • 当循环条件是left<right 进行判断,不能是left<=right(细节二,不然可能会陷入死循环)
  • 当 left == right 时,循环结束,此时 left 或 right 指向目标值的第一个位置(如果存在)
  • 如果 nums[right] != target,说明目标值不存在,直接返回 [-1, -1],否则设置ret[ 0 ] = left,保存开始位置。
     

细节一解析:

求中间位置有两种:

mid = left + (right - left) / 2(数组长度是奇数则在中间位置,偶数长度在中间位置偏左)

mid = left + (right - left + 1) / 2(数组长度是奇数则在中间位置,偶数长度在中间位置偏右)

如果使用left + (right - left + 1) / 2可能会陷入死循环

如果剩下两个数字,mid值大于目标值时,需要更新right = mid位置,相当于没移动,循环判断依旧如此,陷入死循环。


细节二解析:

为什么循环判断条件不能是 left <= right?,一张图解释

当left == right时,还要继续进入循环,还是更新right = mid,不移动,如此循环判断,陷入死循环。

视频展示:

在排序数组中查找元素的第一个和最后一个位置(左端点)

 

 2.3查找右端点(目标值结束位置)

 

1.重新初始化指针:

  • left 指向数组的起始位置(索引 0)。
  • right 指向数组的结束位置(索引 nums.length - 1)。

 

2.二分查找:

  • 计算中间位置 mid = left + (right - left + 1) / 2。(细节一,避免陷入死循环)
  • 如果 nums[mid] <= target,说明目标值在右半部分或当前位置就是目标值,更新 left = mid。
  • 如果 nums[mid] > target,说明目标值在左半部分,更新 right = mid - 1。

 

3.循环退出并判断:

  • 当循环条件是left<right 进行判断,不能是left<=right(细节二,不然可能会陷入死循环)
  • 当 left == right 时,循环结束,此时 left 或 right 指向目标值的最后一个位置(如果存在)。
  • 将 ret[1] 设置为 left,即目标值的结束位置。

细节一解析

如果使用left + (right - left ) / 2可能会陷入死循环

如果剩下两个数字,mid值小于等于目标值时,需要更新left= mid位置,相当于没移动,循环判断依旧如此,陷入死循环。


细节二跟求左端点一致。

 视频展示:

在排序数组中查找元素的第一个和最后一个位置(右端点)

 

三.题目源码

class Solution {
    public int[] searchRange(int[] nums, int target) {
        //判断边界情况
        int[]  ret = new int[2];
        ret[0] = ret [1] = -1;
        if(nums.length == 0) return ret;

        //求左端点
        int left = 0; int right = nums.length - 1;
        while(left < right){
            int mid = left + (right - left ) / 2;
            if(nums[mid] < target) left = mid + 1;
            else right = mid ;
        }
        if(nums[right] != target) return ret;
        else ret[0] = right;

        //求右端点
        left = 0;right = nums.length -1;
        while(left < right){
            int mid = left + (right - left + 1) / 2;
            if(nums[mid] <= target) left = mid;
            else right = mid - 1;

        }
        ret[1] = left;

        return ret;

    }
}


http://www.kler.cn/a/527686.html

相关文章:

  • 【项目集成Husky】
  • 学习串行通信
  • UE5 GAS RPG Character Classes
  • Docker小游戏 | 使用Docker部署2048网页小游戏
  • 《DeepSeek-R1 问世,智能搜索领域迎来新变革》
  • 浅谈AI的发展对IT行业的影响
  • 一文介绍Hive数据类型
  • 寒假刷题Day18
  • Vue.js组件开发-实现滑块滑动无缝切换和平滑切换动画
  • AI作画提示词:Prompts工程技巧与最佳实践
  • 第11章:根据 ShuffleNet V2 迁移学习医学图像分类任务:甲状腺结节检测
  • Java 9模块开发:Eclipse实战指南
  • Autogen_core源码:_agent_runtime.py
  • FOC核心原理的C语言实现
  • Redis代金卷(优惠卷)秒杀案例-单应用版
  • 如何在数据湖中有效治理和管理“数据沼泽”问题,提高数据的可发现性和利用率?
  • vulkan从小白到专家——RenderPassFramebuffer
  • JavaScript函数中this的指向
  • python 文件操作全知道 | python 小知识
  • 36. printf
  • 团体程序设计天梯赛-练习集——L1-029 是不是太胖了
  • 大模型高频知识汇总:查漏补缺参考大全
  • 【Redis】set 和 zset 类型的介绍和常用命令
  • oracl:多表查询>>表连接[内连接,外连接,交叉连接,自连接,自然连接,等值连接和不等值连接]
  • Docker小游戏 | 使用Docker部署跳一跳经典小游戏
  • 23.Word:小王-制作公司战略规划文档❗【5】