forked from HuberTRoy/leetCode
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathFindFirstAndLastPositionOfElementInSortedArray.py
124 lines (89 loc) · 2.73 KB
/
FindFirstAndLastPositionOfElementInSortedArray.py
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
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
"""
Given an array of integers nums sorted in ascending order, find the starting and ending position of a given target value.
Your algorithm's runtime complexity must be in the order of O(log n).
If the target is not found in the array, return [-1, -1].
Example 1:
Input: nums = [5,7,7,8,8,10], target = 8
Output: [3,4]
Example 2:
Input: nums = [5,7,7,8,8,10], target = 6
Output: [-1,-1]
要求: 时间复杂度必须是 O(log n)
看到log n 立马先想到二分,排好序的数组,刚好可以利用二分。
正好 Python 内置模块 bisect 有写二分法,稍加改进即可使用。
一般可能会想到用递归实现,下面是 Python 官方的迭代实现:
```
初始定义两个变量:
lo (lower)
hi (higher)
lo 初始为首位 0.
hi 则为 len(list).
每次都获取 list[mid] mid = (lo + hi) // 2.
list[mid] 有三种情况:
1. 与target相等。
2. 大于target.
3. 小于target.
lo mid hi
1 2 3 4 5 6
若mid大于target则表示要找的目标存在于 lo 与 mid之间。
小于则表示要找的目标存在于 mid 与 hi之间。
要改进的地方在于,处理相等的情况:
在不相等时先移动lo还是hi都无所谓。
在相等时若先移动 lo 则尽可能找到的是最右边的一个。
hi 则是最左边的一个。
lo hi
1 2 2 2 3
先移动 lo
lo hi
1 2 2 2 3
↑
没有可能在找到它。
先移动 hi
lo hi
1 2 2 2 3
↑
没有可能在找到它。
所以一左一右,两次二分即可。
测试地址:
https://leetcode.com/problems/find-first-and-last-position-of-element-in-sorted-array/description/
beat
94%
```
"""
class Solution(object):
def find_right(self, nums, target):
lo = 0
hi = len(nums)
equals = []
while lo < hi:
mid = (lo + hi) // 2
if target == nums[mid]:
equals.append(mid)
if target < nums[mid]:
hi = mid
else:
lo = mid + 1
return equals[-1] if equals else -1
def find_left(self, nums, target):
lo = 0
hi = len(nums)
equals = []
while lo < hi:
mid = (lo + hi) // 2
if target == nums[mid]:
equals.append(mid)
if target > nums[mid]:
lo = mid + 1
else:
hi = mid
return equals[-1] if equals else -1
def searchRange(self, nums, target):
"""
:type nums: List[int]
:type target: int
:rtype: List[int]
"""
left = self.find_left(nums, target)
if left == -1:
return [-1, -1]
return [left, self.find_right(nums, target)]