总体思路:二分法确定哪个区间是增序的。如果确定了一个区间是增序的,则可以在那个区间继续使用二分法缩小范围,如果target不再这个增序区间那么就在另一边。则对另一边不是增序的区间进行第一步再确定那半块里面再细分成的两块哪块是增序区间。
class Solution {
public:
bool search(vector<int>& nums, int target) {
int l = 0, r = nums.size() - 1, mid = 0;
while (l <= r) {
mid = l + (r - l) / 2;
if (nums[mid] == target) return true;
if (nums[l] == nums[mid]) l++; //中点和左端相同,无法判断左区间有序还是右区间有序。
else if (nums[mid] <= nums[r]) { //说明右区间有序增序
if (target > nums[mid] && target <= nums[r]) {
l = mid + 1;
}
else {
r = mid - 1;
}
}
else { // 左区间有序增序
if (target < nums[mid] && target >= nums[l]) {
r = mid - 1;
}
else {
l = mid + 1;
}
}
}
return false;
}
};
注意:需要判断中点是否和左值是否相等,相等时无法确定哪边是有序的,此时只需要左边界加一即可。但为什么中点和右值相等时就不用判断呢???