在排序数组中查找元素的第一个和最后一个位置
题目
在从小到大排序的数组 nums
中, 找出 target
的第一个和最后一个位置的索引的数组.
提示:
0 <= nums.length <= 10⁵
-10⁹ <= nums[i] <= 10⁹
nums
是一个非递减数组-10⁹ <= target <= 10⁹
示例
输入: nums = [6, 7, 8, 8, 10], target = 8
输出: [2, 3]
输入: nums = [1], target = 1
输出: [0, 0]
输入: nums = [1], target = 0
输出: [-1, -1]
题解
先写好二分查找的基本框架, 然后考察几个特例:
nums = [4, 6, 8, 9, 10], target = 6
, 此时nums[mid]
为8
, 大于target
, 那必然right = mid - 1
nums = [4, 6, 8, 9, 10], target = 9
, 此时nums[mid]
为8
, 小于target
, 那必然left = mid + 1
nums = [4, 6, 8, 8, 9, 10], target = 8
, 此时nums[mid]
为8
, 等于target
, 但我们要获取的是一个区间范围, 因此:- 如果
nums[left]
小于了target
, 那就尝试让left
往右走一步 - 如果
nums[right]
大于了target
, 那就尝试让right
往左走一步 - 如果
nums[left]
和nums[right]
同时都等于了target
, 我们也就找到了这个区间范围, 返回即可
- 如果
nums = [2], target = 0
, 此时nums[mid]
为2
, 大于target
, 那必然right = mid - 1
, 而此时right
就小于了left
, 退出循环, 返回[-1, -1]
.nums = [2], target = 5
, 此时nums[mid]
为2
, 小于target
, 那必然left = mid + 1
, 而此时left
就大于了right
, 退出循环, 返回[-1, -1]
.
- JavaScript
- Rust
/**
* @param {number[]} nums
* @param {number} target
* @return {number[]}
*/
var searchRange = function (nums, target) {
let left = 0
let right = nums.length - 1
while (left <= right) {
const mid = ((left + right) / 2) | 0
if (nums[mid] === target) {
if (nums[left] < target) left++
if (nums[right] > target) right--
if (nums[left] === target && nums[right] === target) {
return [left, right]
}
} else if (nums[mid] < target) {
left = mid + 1
} else if (nums[mid] > target) {
right = mid - 1
}
}
return [-1, -1]
}
pub fn search_range(nums: Vec<i32>, target: i32) -> Vec<i32> {
let (mut left, mut right) = (0, nums.len() - 1);
while left <= right {
// 注意下面 right -= 1 和 right = mid - 1 都有可能将 right 变为负数,
// 由于 right 是 uszie 类型, 因此一旦它变成负数, 会自动被转化成 usize::MAX,
// 这样后面的迭代中就会溢出, 需要过滤到这种情况
if (right > nums.len()) {
break;
}
let mid = (left + right) / 2;
if nums[mid] == target {
if nums[left] < target {
left += 1;
}
if nums[right] > target {
right -= 1;
}
if nums[left] == target && nums[right] == target {
return vec![left as i32, right as i32];
}
} else if nums[mid] < target {
left = mid + 1;
} else if nums[mid] > target {
right = mid - 1;
}
}
vec![-1, -1]
}