欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 健康 > 养生 > 在排序数组中查找元素的第一个和最后一个位置(力扣)

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

2025/2/3 23:52:02 来源:https://blog.csdn.net/LHY537200/article/details/145399627  浏览:    关键词:在排序数组中查找元素的第一个和最后一个位置(力扣)

一.题目介绍

二.题目解析

使用二分进行查找

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;}
}

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com