题目描述
解题思路
二分法
排序数组中的搜索问题,首先想到 二分法 解决。
排序数组 nums 中的所有数字 target 形成一个窗口,记窗口的 左 / 右边界 索引分别为 left 和 right ,分别对应窗口左边 / 右边的首个元素。
本题要求统计数字 target 的出现次数,可转化为:使用二分法分别找到 左边界 left 和 右边界 right ,易得数字 target 的数量为 right - left - 1。
算法解析:
- 初始化: 左边界 i = 0 ,右边界 j = len(nums) - 1。
- 循环二分: 当闭区间 [i, j][i,j] 无元素时跳出;
- 计算中点 m = (i + j) / 2m=(i+j)/2 (向下取整);
- 若 nums[m] < target ,则 target 在闭区间 [m + 1, j]中,因此执行 i = m + 1;
- 若 nums[m] > target ,则 target 在闭区间 [i, m - 1] 中,因此执行 j = m - 1;
- 若 nums[m] = target ,则右边界 right在闭区间 [m+1, j]中;左边界 left在闭区间 [i, m-1] 中。因此分为以下两种情况:
- 若查找 右边界 right,则执行 i = m + 1 ;(跳出时 ii 指向右边界)
- 若查找 左边界 left,则执行 j = m - 1;(跳出时 jj 指向左边界)
- 返回值: 应用两次二分,分别查找 right和 left,最终返回 right - left - 1 即可。
效率优化:
以下优化基于:查找完右边界 right = iright=i 后,则 nums[j]nums[j] 指向最右边的 targettarget (若存在)。
查找完右边界后,可用 nums[j] = target 判断数组中是否包含 target,若不包含则直接提前返回 0 ,无需后续查找左边界。
查找完右边界后,左边界 left 一定在闭区间 [0, j]中,因此直接从此区间开始二分查找即可。
也就是通过2次二分法查找左右边界,在将左右边界进行计算。
/**
* 二分法
*
* @param nums
* @param target
* @return
*/
public int search(int[] nums, int target) {
if (nums.length == 0) {
return 0;
}
int left = 0;
int right = nums.length - 1;
// 搜索有边界
while (left <= right) {
int mid = (left + right) / 2;
if (target >= nums[mid]) {
left = mid + 1;
} else right = mid - 1;
}
int edgeRight = left; // 获取到右边界
// 若数组中无 target ,则提前返回
if (right > 0 && nums[right] != target) return 0;
left = 0;
right = nums.length - 1;
while (left <= right) {
int mid = (left + right) / 2;
if (target > nums[mid]) {
left = mid + 1;
} else right = mid - 1;
}
int edgeLeft = right;
return edgeRight - edgeLeft - 1;
}
- 时间复杂度 O(log N) : 二分法为对数级别复杂度。
- ** : 几个变量使用常数大小的额外空间。
哈希表
/**
* 哈希表
* @param nums
* @param target
* @return
*/
public int search(int[] nums, int target) {
HashMap<Integer, Integer> map = new HashMap<>();
for (int i = 0; i < nums.length; i++) {
map.put(nums[i], map.getOrDefault(nums[i], 0) + 1);
}
return map.get(target) == null ? 0 : Integer.valueOf(map.get(target));
}