题目描述
把一个数组最开始的若干个元素搬到数组的末尾,我们称之为数组的旋转。输入一个递增排序的数组的一个旋转,输出旋转数组的最小元素。例如数组{3,4,5,1,2}为{1,2,3,4,5}的一个旋转,该数组的最小值为1。
题解思路
从头到尾遍历数组一次,我们就能找出最小的元素。这种思路的时间复杂度显然是O(n)。但是这个思路没有利用输入的旋转数组的特性,肯定达不到面试官的要求。
旋转之后的数组实际上可以划分为两个排序的子数组,而且前面的子数组的元素都大于或者等于后面子数组的元素。我们还注意到最小的元素刚好是这两个子数组的分界线。在排序的数组中我们可以用二分查找法实现O(logn)的查找。
-
和二分查找法一样,我们用两个指针分别指向数组的第一个元素和最后一个元素。
-
接着我们可以找到数组中间的元素:
- 如果该中间元素位于前面的递增子数组,那么它应该大于或者等于第一个指针指向的元素。此时数组中最小的元素应该位于该中间元素的后面。我们可以把第一个指针指向该中间元素,这样可以缩小寻找的范围。移动之后的第一个指针仍然位于前面的递增子数组之中。
- 如果中间元素位于后面的递增子数组,那么它应该小于或者等于第二个指针指向的元素。此时该数组中最小的元素应该位于该中间元素的前面。
-
接下来我们再用更新之后的两个指针,重复做新一轮的查找。
按照上述的思路,第一个指针总是指向前面递增数组的元素,而第二个指针总是指向后面递增数组的元素。最终第一个指针将指向前面子数组的最后一个元素,而第二个指针会指向后面子数组的第一个元素。也就是它们最终会指向两个相邻的元素,而第二个指针指向的刚好是最小的元素。这就是循环结束的条件。
以前面的数组{3,4,5,1,2}为例,下图展示了在该数组中查找最小值的过程:
代码实现
int MinInOrder(vector<int> v, int i, int j){
int res = v[i];
for(int n = i+1; n <= j; n++){
if(res > v[n]) res = v[n];
}
return res;
}
int Min(vector<int> v, int len){
if(len <= 0) {
cout << "Invaild parameters" << endl;
return -1;
}
int i = 0, j = len-1, mid = 0;
while(v[i] >= v[j]){
if(j-i == 1){
mid = j;
break;
}
mid = i + (j - i)/2;
if(v[i] == v[mid] && v[mid] == v[j]) return MinInOrder(v, i, j);
if(v[i] <= v[mid]) i = mid;
else if(v[j] >= v[mid]) j = mid;
}
return v[mid];
}
这里需要注意的是:
-
把 mid 初始化为 i 的原因:一旦发现数组中第一个数字小于最后一个数字,表明该数组是排序的,就可以直接返回第一个数字了。
-
特殊情况的分析:如果下标为 i 、j 和 mid 指向的三个数字相等,则只能顺序查找,因此这里定义了一个MinInOrder()方法。
测试接口
#include <iostream>
#include <vector>
using namespace std;
int main(){
int arr[] = { 1, 1, 1, 0, 1};
/*
int arr[] = { 1, 0, 1, 1, 1};
int arr[] = { 1, 2, 3, 4, 5};
int arr[] = { 3, 4, 5, 1, 2};
*/
vector<int> v(arr, arr+5);
int sz = v.size();
cout << Min(v, sz) << endl;
return 0;
}
如有不同见解,欢迎留言讨论~~~