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

news/2025/2/5 13:33:31/

一.题目介绍

二.题目解析

使用二分进行查找

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.ppmy.cn/news/1569515.html

相关文章

240. 搜索二维矩阵||

参考题解&#xff1a;https://leetcode.cn/problems/search-a-2d-matrix-ii/solutions/2361487/240-sou-suo-er-wei-ju-zhen-iitan-xin-qin-7mtf 将矩阵旋转45度&#xff0c;可以看作一个二叉搜索树。 假设以左下角元素为根结点&#xff0c; 当target比root大的时候&#xff…

用BGP的路由聚合功能聚合大陆路由,效果显著不?

正文共&#xff1a;666 字 11 图&#xff0c;预估阅读时间&#xff1a;1 分钟 之前我们统计过中国境内的IP地址和路由信息&#xff08;你知道中国大陆一共有多少IPv4地址吗&#xff1f;&#xff09;&#xff0c;不过数量比较多&#xff0c;有8000多条。截止到2021年底&#xff…

【Leetcode刷题记录】1456. 定长子串中元音的最大数目---定长滑动窗口即解题思路总结

1456. 定长子串中元音的最大数目 给你字符串 s 和整数 k 。请返回字符串 s 中长度为 k 的单个子字符串中可能包含的最大元音字母数。 英文中的 元音字母 为&#xff08;a, e, i, o, u&#xff09;。 这道题的暴力求解的思路是通过遍历字符串 s 的每一个长度为 k 的子串&#xf…

生成式AI安全最佳实践 - 抵御OWASP Top 10攻击 (上)

今天小李哥将开启全新的技术分享系列&#xff0c;为大家介绍生成式AI的安全解决方案设计方法和最佳实践。近年来&#xff0c;生成式 AI 安全市场正迅速发展。据 IDC 预测&#xff0c;到 2025 年全球 AI 安全解决方案市场规模将突破 200 亿美元&#xff0c;年复合增长率超过 30%…

什么叫DeepSeek-V3,以及与GPT-4o的区别

1. DeepSeek 的故事 1.1 DeepSeek 是什么&#xff1f; DeepSeek 是一家专注于人工智能技术研发的公司&#xff0c;致力于打造高性能、低成本的 AI 模型。它的目标是让 AI 技术更加普惠&#xff0c;让更多人能够用上强大的 AI 工具。 1.2 DeepSeek-V3 的问世 DeepSeek-V3 是…

PyQt4学习笔记2】Qt 的 Model/View 架构

目录 一、Model/View 架构 1. Model/View 架构的核心思想 2. Model/View 架构的优势 3. 常见的 Model 和 View 类 模型类 (Model) 视图类 (View) 4. 信号和槽机制 5. 示例代码 6. 数据操作 设置数据(将特定位置的数据替换掉) 插入和删除数据 7. 自定义模型 8. 总结 二、Model/V…

传输层协议 UDP 与 TCP

&#x1f308; 个人主页&#xff1a;Zfox_ &#x1f525; 系列专栏&#xff1a;Linux 目录 一&#xff1a;&#x1f525; 前置复盘&#x1f98b; 传输层&#x1f98b; 再谈端口号&#x1f98b; 端口号范围划分&#x1f98b; 认识知名端口号 (Well-Know Port Number) 二&#xf…

[LeetCode] 字符串完整版 — 双指针法 | KMP

字符串 基础知识双指针法344# 反转字符串541# 反转字符串II54K 替换数字151# 反转字符串中的单词55K 右旋字符串 KMP 字符串匹配算法28# 找出字符串中第一个匹配项的下标#459 重复的子字符串 基础知识 字符串的结尾&#xff1a;空终止字符00 char* name "hello"; …