题目描述
把一个数组最开始的若干个元素搬到数组的末尾,我们称之为数组的旋转。输入一个递增排序的数组的一个旋转,输出旋转数组的最小元素。例如,数组 [3,4,5,1,2] 为 [1,2,3,4,5] 的一个旋转,该数组的最小值为1。
示例 :
输入:[2,2,2,0,1]
输出:0
来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/xuan-zhuan-shu-zu-de-zui-xiao-shu-zi-lcof
解题思路
我们要注意读题, 旋转数组旋转之前是有序的,所以旋转之后,最小值的左右两边的数都是有序的
从头到尾遍历数组一次,我们就能找出最小的元素。这种思路的时间复杂度显然是O(n)。
我们注意到旋转之后的数组实际上可以划分为两个排序的子数组,而且前面的子数组的元素都大于或者等于后面子数组的元素。我们还注意到最小的元素刚好是这两个子数组的分界线。在排序的数组中我们可以用二分查找法实现O(logn)的查找
Step1.和二分查找法一样,我们用两个指针分别指向数组的第一个元素和最后一个元素。
Step2.接着我们可以找到数组中间的元素:
如果该中间元素位于前面的递增子数组,那么它应该大于或者等于第一个指针指向的元素。此时数组中最小的元素应该位于该中间元素的后面。我们可以把第一个指针指向该中间元素,这样可以缩小寻找的范围。移动之后的第一个指针仍然位于前面的递增子数组之中。如果中间元素位于后面的递增子数组,那么它应该小于或者等于第二个指针指向的元素。此时该数组中最小的元素应该位于该中间元素的前面。
Step3.接下来我们再用更新之后的两个指针,重复做新一轮的查找。
按照上述的思路,第一个指针总是指向前面递增数组的元素,而第二个指针总是指向后面递增数组的元素。最终第一个指针将指向前面子数组的最后一个元素,而第二个指针会指向后面子数组的第一个元素。也就是它们最终会指向两个相邻的元素,而第二个指针指向的刚好是最小的元素。这就是循环结束的条件。
参考
剑指Offer面试题:7.旋转数组的最小数字
漫画:“旋转数组”中的二分查找
public static int GetMin(int[] numbers)
{
if (numbers == null || numbers.Length <= 0)
{
return int.MinValue;
}
int index1 = 0;
int index2 = numbers.Length - 1;
// 把indexMid初始化为index1的原因:
// 一旦发现数组中第一个数字小于最后一个数字,表明该数组是排序的
// 就可以直接返回第一个数字了
int indexMid = index1;
while (numbers[index1] >= numbers[index2])
{
// 如果index1和index2指向相邻的两个数,
// 则index1指向第一个递增子数组的最后一个数字,
// index2指向第二个子数组的第一个数字,也就是数组中的最小数字
if (index2 - index1 == 1)
{
indexMid = index2;
break;
}
indexMid = (index1 + index2) / 2;
// 特殊情况:如果下标为index1、index2和indexMid指向的三个数字相等,则只能顺序查找
if (numbers[index1] == numbers[indexMid] && numbers[indexMid] == numbers[index2])
{
return GetMinInOrder(numbers, index1, index2);
}
// 缩小查找范围
if (numbers[indexMid] >= numbers[index1])
{
index1 = indexMid;
}
else if (numbers[indexMid] <= numbers[index2])
{
index2 = indexMid;
}
}
return numbers[indexMid];
}
public static int GetMinInOrder(int[] numbers, int index1, int index2)
{
int result = numbers[index1];
for (int i = index1 + 1; i <= index2; ++i)
{
if (result > numbers[i])
{
result = numbers[i];
}
}
return result;
}
1逐个查找
public int minArray(int[] numbers) {
int min = numbers[0];
for (int i = 1; i < numbers.length; i++) {
if (min > numbers[i])
min = numbers[i];
}
return min;
}
作者:sdwwld
链接:https://leetcode-cn.com/problems/xuan-zhuan-shu-zu-de-zui-xiao-shu-zi-lcof/solution/liang-chong-jie-jue-fang-shi-du-ji-bai-liao-100de-/
来源:力扣(LeetCode)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
2 排序查找
public int minArray(int[] numbers) {
Arrays.sort(numbers);
return numbers[0];
}
作者:sdwwld
链接:https://leetcode-cn.com/problems/xuan-zhuan-shu-zu-de-zui-xiao-shu-zi-lcof/solution/liang-chong-jie-jue-fang-shi-du-ji-bai-liao-100de-/
来源:力扣(LeetCode)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
二分法查找
public int minArray(int[] numbers) {
int left = 0, right = numbers.length - 1;
while (left < right) {
//找出left和right中间值的索引
int mid = left + (right - left) / 2;//因为low+high在low和high特别大的时候可能会造成溢出,使用减法避免了溢出发生
if (numbers[mid] > numbers[right]) {
//如果中间值大于最右边的值,说明旋转之后最小的
//数字肯定在mid的右边,比如[3, 4, 5, 6, 7, 1, 2]
left = mid + 1;
} else if (numbers[mid] < numbers[right]) {
//如果中间值小于最右边的值,说明旋转之后最小的
//数字肯定在mid的前面,比如[6, 7, 1, 2, 3, 4, 5],
//注意这里mid是不能减1的,比如[3,1,3],我们这里只是
//证明了numbers[mid]比numbers[right]小,但有可能
//numbers[mid]是最小的,所以我们不能把它给排除掉
right = mid;
} else {
//如果中间值等于最后一个元素的值,我们是没法确定最小值是
// 在mid的前面还是后面,但我们可以缩小查找范围,让right
// 减1,因为即使right指向的是最小值,但因为他的值和mid
// 指向的一样,我们这里并没有排除mid,所以结果是不会有影响的。
//比如[3,1,3,3,3,3,3]和[3,3,3,3,3,1,3],中间的值
//等于最右边的值,但我们没法确定最小值是在左边还是右边
right--;
}
}
return numbers[left];
}
作者:sdwwld
链接:https://leetcode-cn.com/problems/xuan-zhuan-shu-zu-de-zui-xiao-shu-zi-lcof/solution/liang-chong-jie-jue-fang-shi-du-ji-bai-liao-100de-/
来源:力扣(LeetCode)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。