LeetCode-面试题53-1-在排序数组中查找数字I
# LeetCode-面试题53-1-在排序数组中查找数字I
统计一个数字在排序数组中出现的次数。
示例1:
输入: nums = [5,7,7,8,8,10], target = 8
输出: 2
1
2
2
1
2
2
示例2:
输入: nums = [5,7,7,8,8,10], target = 6
输出: 0
1
2
2
1
2
2
限制:
0 <= 数组长度 <= 50000
# 解题思路1
在有序的数组中二分查找,确定第一个k出现的位置和最后一个k出现的位置,然后两个位置相减即是出现次数
# 解题思路2
hash表,遍历的过程中把次数加上去即可,速度慢于2分查找
# Java代码
class Solution {
public int search(int[] nums, int target) {
int len = nums.length;
int count = 0;
if(nums!=null&&len>0){
int first = GetFristK(nums,len,target,0,len-1);
int last = GetLastK(nums,len,target,0,len-1);
if(first>-1&&last>-1){
count = last-first+1;
}
}
return count;
}
public int GetFristK(int[] nums,int len,int target,int start,int end){
if(start>end)
return -1;
int mid = (start+end)/2;
int midData = nums[mid];
if(midData==target){
// 找到第一个k的位置
if(mid>0&&nums[mid-1]!=target||mid==0)
return mid;
else // 如果前面还有k,缩小范围继续找
end = mid-1;
}else if(midData>target)
end = mid-1;
else
start = mid+1;
return GetFristK(nums,len,target,start,end);
}
public int GetLastK(int[] nums,int len,int target,int start,int end){
if(start>end)
return -1;
int mid = (start+end)/2;
int midData = nums[mid];
if(midData==target){
// 找到最后一个k的位置
if(mid<len-1&&nums[mid+1]!=target||mid==len-1)
return mid;
else
start = mid+1;
}else if(midData<target)
start = mid+1;
else
end = mid-1;
return GetLastK(nums,len,target,start,end);
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
# Java代码2
class Solution {
public int search(int[] nums, int target) {
int len = nums.length;
if (len == 0) return 0;
int left = 0;
int right = len-1;
// 通过控制等号控制是左边界还是右边界
while(left<=right){
int mid = left+(right-left)/2;
// 当小于等于target时一直移动左指针
if(nums[mid]<=target){
left = mid+1;
} else { // 当大于target时才移动右指针,这样保障了右指针指向重复target的最后一个位置
right = mid-1;
}
}
int rightIndex = right;
left = 0;
right = len-1;
while(left<=right){
int mid = left+(right-left)/2;
// 当大于等于target时一直移动右指针收缩
if(nums[mid]>=target){
right = mid-1;
} else { // 当小于target时才移动左指针,保障左指针指向重复target的第一个位置
left = mid+1;
}
}
int leftIndex = left;
return rightIndex-leftIndex+1;
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
# Java代码3
class Solution {
public int search(int[] nums, int target) {
HashMap<Integer,Integer> map = new HashMap<>();
for(int i : nums){
map.put(i,map.getOrDefault(i,0)+1);
}
return map.containsKey(target)?map.get(target):0;
}
}
1
2
3
4
5
6
7
8
9
2
3
4
5
6
7
8
9
1
2
3
4
5
6
7
8
9
2
3
4
5
6
7
8
9
编辑 (opens new window)
上次更新: 2022/11/18, 11:15:10
- 01
- SpringCache基本配置类05-16
- 03
- Rpamis-security-原理解析12-13