面试题 10.03. 搜索旋转数组
来源: LeetCode 面试题 10.03. 搜索旋转数组
题目描述
面试题 10.03. 搜索旋转数组
搜索旋转数组。给定一个排序后的数组,包含n个整数,但这个数组已被旋转过很多次了,次数不详。请编写代码找出数组中的某个元素,假设数组元素原先是按升序排列的。若有多个相同元素,返回索引值最小的一个。
示例1:
输入: arr = [15, 16, 19, 20, 25, 1, 3, 4, 5, 7, 10, 14], target = 5
输出: 8(元素5在该数组中的索引)
示例2:
输入:arr = [15, 16, 19, 20, 25, 1, 3, 4, 5, 7, 10, 14], target = 11
输出:-1 (没有找到)
提示:
arr 长度范围在[1, 1000000]之间
思路分析
打眼一瞅直接暴力
代码
class Solution {
public:
int search(vector<int>& arr, int target) {
// vilonce
for(int i=0; i<arr.size(); ++i){
if(arr[i] == target) return i;
}
return -1;
}
};
算法分析
- 时间复杂度:O(n)
- 空间复杂度:O(1)
代码改进
说实话感觉这题真的很蠢,为出而出
可以用二分法搜索判断旋转点,用二分法结合之前确定的nums[mid]与旋转点的关系确定如何移动,如果不能确定那就分治解决
关键是要细节考虑很多情况最后实际运行时间并不一定降低多少(小数据)
class Solution {
public:
int search(vector<int>& arr, int target) {
int left = 0, right = arr.size() - 1;
while (left < right) {
int mid = left + (right - left) / 2;
if (arr[mid] > arr[left]) {
if (arr[left] <= target && target <= arr[mid]) {
right = mid;
} else {
left = mid + 1;
}
} else if(arr[mid] < arr[left]){
if (arr[left] <= target || target <= arr[mid]) { // 如果目标在左边,右边界移动到mid
right = mid;
} else { // 否则目标在右半边,左边界移动到mid+1
left = mid + 1;
}
} else {
if (arr[left] == target) {
return left;
} else {
left ++;
}
}
}
return arr[left] == target ? left : -1;
}
};
**********************************************************
相似扩展
归纳总结