1. 分析题目
(1)升序:意味着不必遍历,类似于一本词典按照首字母的顺序排列,可以通过二分法减少搜寻时间。
- 遍历的搜寻时间复杂度是O(n)
- 二分法的时间复杂度是O(log n)
(2)n的范围意味什么?意味着这是一个可以很大的数组,遍历会导致可能的时间很长,这是一个暗示,暗示你需要去考虑时间复杂度和算法的选择。可能要考虑溢出。
(3)nums元素的范围意味着什么?暂时没看出。
2. 解题思路
框架构建:如果存在目标值,返回下标,否则返回-1。可以通过两块来实现:
(1)搜索目标,找到后返回下标。使用while循环,这是是一个排除不符合元素,缩小区间的过程,直到所有元素都被排除。我习惯使用闭区间[left, right]。根据target与中值大小的比较,若相等,则跳出循环。否则,需要将左右端点进行调整。调整的位置是原中值mid的左一或右一位置,由此排除原来的中值mid。直到到左端点大于右端点,此时所有元素都被排除。跳出循环。因此while循环的判断条件是“左端点大于右端点”的否命题——left<=right。
(2)没有找到target 后跳出循环,返回-1。
3. 代码
class Solution {
public:
int search(vector<int>& nums, int target) {
int right = nums.size()-1;
int left = 0;
while (left <= right){
int mid = left + ((right - left) >> 1);
// int mid = (left + right) / 2;
// int mid = left + (right - left)/2
if (nums[mid] < target){
left = mid + 1;
} else if (nums[mid] > target){
right = mid - 1;
} else {
return mid;
}
}
return -1;
}
};
tips:
1. 防止溢出
虽然该题目的n不是特别大,不会出现溢出现象。但是如果没有提示2,防止溢出部分必须写上。
int mid = left + ((right - left) >> 1);
int mid = left + (right - left)/2
2. 位运算进行优化,加快速度
向右位移一位相当于除以2,但是比起除运算,速度更快。
int mid = left + ((right - left) >> 1);