题目:把一个数组最开始的若干个元素搬到数组的末尾,我们称之为数组的旋转。输入一个递增排序的数组的一个旋转,输出旋转数组的最小元素。例如数组{3,4,5,1,2}为{1,2,3,4,5}的一个旋转,该数组的最小值为1.
思路:旋转之后的数组实际上可以划分成两个排序的子数组,最小数是两个子树组的分界线,试着用二分查找的方法来寻找最小数。
使用指针指向数组的第一个元素和最后一个元素,求两个指针的中值,如果中值大于等于前面指针指向的元素,那么说明中值处在前面的递增子数组中,将第一个指针指向向中值的位置;如果中值小于等于第二个指针指向的值,说明中值处在后面递增子数组中,将第二个指针更新为中值所在位置。
程序的终止条件:第一个指针总是指向前面递增数组的元素,而第二个指针总是指向后面递增数组的元素。最终第一个指针指向前面子数组的最后一个元素,而第二个指针指向后面子数组的第一个元素,也就是它们会最终指向两个相邻的元素,而第二个指针指向的刚好是最小的元素。
特殊情况:如果中值同时等于两个指针指向的值,那么指针该如何移动呢?别无他法,最笨的办法——顺序查找。
int min(int* numbers, int length)
{
if(numbers == NULL || length <= 0)
throw new std::exception("Invalid parameters");
int index1 = 0;
int index2 = length - 1;
int indexMid = index1;
while(numbers[index1] >= numbers[index2])
{
if(index2 - index1 == 1)
{
indexMid = index2;
break;
}
indexMid = (index1 + index2) /2;
if(numbers[index1] == numbers[index2] && numbers[indexMid] == numbers[index1])
return MinInOrder(numbers, index1, index2);
if(numbers[indexMid] >= numbers[index1]
index1 = indexMid;
else if(numbers[indexMid] <= numbers[index2])
index2 = indexMid;
}
return numbers[indexMid];
}
int MinInOrder(int* numbers, int index1, int index2)
{
int result = numbers[index1];
for(int i = index + 1; i <= index2; ++i)
{
if(result > numbers[i]
result = numbers[i];
}
return result;
}