Question

Suppose an array sorted in ascending order is rotated at some pivot unknown to you beforehand.

(i.e.,  [0,1,2,4,5,6,7] might become  [4,5,6,7,0,1,2]).

Find the minimum element.

Answer

借用以下网上的翻译:

  把一个数组最开始的若干个元素搬到数组的末尾,我们称之为数组的旋转。 输入一个非减排序的数组的一个旋转,输出旋转数组的最小元素。 例如数组{3,4,5,1,2}为{1,2,3,4,5}的一个旋转,该数组的最小值为1。 NOTE:给出的所有元素都大于0,若数组大小为0,请返回0。

  下面我们来解题:

  最初可以通过画图理解,我想或许我们可以画着以下的图(为了方便,离散点画成连续的):

  当然这个草图也真是草图,很粗糙,哈哈,而且不能解释所有情况,它只能解决严格递增。不过,这应该能解决大部分情况,至于其它小部分的情况我们等下再考虑。

主要思路

  我们头脑第一个蹦出来的想法或许就是遍历一次取最小值就好,这很简单,也很容易实现,但这显然不是我们想要的,时间复杂度为O(n)。

  我们想另外一种方法。非减?有序?有没有觉得有点像二分查找,只不过它被分成两截了,不过这并不能难倒我们,我们不妨试着用这思路解决。

  相对于二分查找,我们没有所要查找的目标,数组也不是完全有序。不过,我们想想如何模仿这种过程,二分查找是取中间值来跟目标值比较进而缩小范围的,而我们这题如何缩小范围呢,取什么比较呢,如果取中间值,又和谁比较呢。靠直觉,我们取最初点(设为A)、最后一个点(设为C),与中间点(设为B)比较,因为这三个点最具代表性,对于A与C来说,很显然A>=C,我们再分析B,B可能在左半边直线上,也有可能在右半边直线;如果在左半边直线,这时候B>=A>=C,这时候可以断定最小值在B-C之间,因为B-C不是连续递增的;如果在右半边直线,这时候A>=C>=B,可以断定最小值在A-B之间,因为A-B之间不是连续递增的,肯定在某个地方有个“坑”。总结一下,如果B>C,就继续在B-C查找,如果A>B,则在A-C查找。

  我们还需要个终止条件,就是让这个查找终止下来。我们通过上面的算法不断缩小搜索范围,最终肯定缩小到两个元素,因为如果存在三个元素以上,中间值是取除边界两个点(A,C)外的其中一个点,这样下次搜索范围能继续缩小,直到两个点。最后我们需要得到最小值,而两个元素必然存在着最小值,但哪个是最小值呢,为了避免麻烦,我们可以比较一次得出最小值。

 (括号可以不看,单纯说下我最初考虑两个值种取最小值的过程,A与C之间的范围逐渐缩小,逐渐逼近最小值,最后剩下的两个点的分布有两种可能,一种是分别在一条直线上,另一种可能是都在右半条直线上,然后,我们细想这个逼近的过程是可以排除第二种可能的,因为点A是不会随着搜索范围的减少而到达下面那条直线的。因此,只会分别在一条直线上,从图上来看两个点恰好是一上一下,这样我们取下面那点,就是那个数组下标偏大的那个点即可,不过这会在我们没有讨论过的一些特例里失效,比如旋转数组为[1,2],因为这不符合我们上面讨论的模型,因为[1,2]全部旋转了,回到最初的位置,元素位置并没有变化,因此我们采用比较两个元素的方法得到最小值比较好)

  所以我们能像二分查找那样写出大致以下的代码:

    int findMinVal(int min, int max, const vector<int> &rotateArray) {
// 取三个点的值
int minVal = rotateArray[min]; //最左边的点的值
int maxVal = rotateArray[max]; //最右边的点的值
int midVal = rotateArray[(min + max) / ]; //中间的点的值 // 终止条件
      if ((max - min) <= )
        return minVal > maxVal ? maxVal : minVal; // 根据三个点的值缩小搜索范围
if (midVal > maxVal)
return findMinVal((min + max) / , max, rotateArray);
else if (minVal > midVal)
return findMinVal(min, (min + max) / , rotateArray); // 先无视这个return先,等等再讨论
return minVal;
}

考虑特殊情况

  如果我们放上Leetcode提交,这肯定是过不了的。如果你是个谨慎的人,一定会想到有各种特殊情况,特别是三个点的值之间存在相等关系的时候。我们下面就来考虑这些情况。

  首先是如果midVal > maxVal或者minVal > midVal,那么在所有情况下都是能正确的缩小搜索范围的,这就不讨论了。
  所以我们剩下要讨论的情况是midVal <= maxVal 且 minVal <= midVal,组合起来就是minVal <= midVal <= maxVal。我们分四种情况考虑 。

  ①minVal < midVal < maxVal,这种情况存在于像[1,2,3]完全旋转后和最初数组一样的数组。这种很明显取minVal即可。

②minVal < midVal = maxVal,这种和①一样,例如数组为[1,2,2],取minVal即可。

③minVal = midVal < maxVal,  同①,例如[1,1,2],取minVal即可。

④minVal = midVal = maxVal,  这种情况有些特殊,不能缩小一半的范围,也不能得出最小值,例如[1,1,0,1]或[1,1,0,1,1,1,1],所以只能一步一步缩小,即将索引min+1,max-1。

  ①-③可以合并。最终代码如下:

  

    int findMin(vector<int>& nums) {
if (nums.empty())
return ;
return findMinVal(, nums.size() - , nums);
} int findMinVal(int min, int max, const vector<int> &rotateArray) { int minVal = rotateArray[min];
int maxVal = rotateArray[max];
int midVal = rotateArray[(min + max) / ];
if ((max - min) <= )
return minVal > maxVal ? maxVal : minVal;
if (midVal > maxVal)
return findMinVal((min + max) / , max, rotateArray);
else if (minVal > midVal)
return findMinVal(min, (min + max) / , rotateArray);
else if (minVal == midVal && midVal == maxVal)
return findMinVal(min+1, max-1, rotateArray);
return
minVal;
}

  由于上面的讨论缺乏严格完整的数学证明过程,我不敢保证能考虑到所有情况,如果有漏了某些情况,请告诉我= =,哈哈。不过是能通过leetcode和牛客网的检测的。

Leetcode Week4 Find Minimum in Rotated Sorted Array II的更多相关文章

  1. [LeetCode] 154. Find Minimum in Rotated Sorted Array II 寻找旋转有序数组的最小值 II

    Follow up for "Find Minimum in Rotated Sorted Array":What if duplicates are allowed? Would ...

  2. Java for LeetCode 154 Find Minimum in Rotated Sorted Array II

    Suppose a sorted array is rotated at some pivot unknown to you beforehand. (i.e., 0 1 2 4 5 6 7 migh ...

  3. 【leetcode】Find Minimum in Rotated Sorted Array II JAVA实现

    一.题目描述 Follow up for "Find Minimum in Rotated Sorted Array":What if duplicates are allowed ...

  4. [LeetCode] 154. Find Minimum in Rotated Sorted Array II 寻找旋转有序数组的最小值之二

      Suppose an array sorted in ascending order is rotated at some pivot unknown to you beforehand. (i. ...

  5. leetcode 154. Find Minimum in Rotated Sorted Array II --------- java

    Follow up for "Find Minimum in Rotated Sorted Array":What if duplicates are allowed? Would ...

  6. [LeetCode#154]Find Minimum in Rotated Sorted Array II

    The question: Follow up for "Find Minimum in Rotated Sorted Array":What if duplicates are ...

  7. leetcode 【 Find Minimum in Rotated Sorted Array II 】python 实现

    题目: Follow up for "Find Minimum in Rotated Sorted Array":What if duplicates are allowed? W ...

  8. LeetCode 154. Find Minimum in Rotated Sorted Array II寻找旋转排序数组中的最小值 II (C++)

    题目: Suppose an array sorted in ascending order is rotated at some pivot unknown to you beforehand. ( ...

  9. LeetCode 154.Find Minimum in Rotated Sorted Array II(H)(P)

    题目: Suppose an array sorted in ascending order is rotated at some pivot unknown to you beforehand. ( ...

随机推荐

  1. vim 快捷键方式

    https://juejin.im/post/5ab1275d5188255588053e70#heading-14 安装方式 https://juejin.im/entry/57b281f72e95 ...

  2. Head First设计模式——代理模式

    在HeadFirst设计模式中代理模式用了比较多的篇幅来讲解,其中的例子我感觉有些繁琐,所以我们这篇就不按照惯例用例子来阐述代理模式了.我们直接进入正题,分析模式本身的设计和解决的问题. 远程代理模式 ...

  3. cesium结合geoserver实现地图空间查询(附源码下载)

    前言 cesium 官网的api文档介绍地址cesium官网api,里面详细的介绍 cesium 各个类的介绍,还有就是在线例子:cesium 官网在线例子,这个也是学习 cesium 的好素材. 内 ...

  4. Android中使用getDrawable时提示:Call requires API level 21(current min is 15)

    场景 在通过getDrawable方法获取照片资源时提示: Call requires API level 21(current min is 15) 注: 博客: https://blog.csdn ...

  5. 记一次Postgres CPU爆满故障

    问题描述 公司项目测试环境调用某些接口的时候,服务器立即崩溃,并一定时间内无法提供服务. 问题排查 服务器配置不够 第一反应是服务器需要升配啦,花钱解决一切!毕竟测试服务器配置确实不高,2CPU + ...

  6. RabbitMQ安装(发生系统错误5。拒绝访问。发生系统错误1067。进程意外终止。)

    RabbitMQ安装步骤(windows) 1.RabbitMQ建立在强大的Erlang OTP平台上,因此我们首先需要安装Erlang. 2.接下来安装RabbitMQ服务rabbitmq-serv ...

  7. 软件模拟spi的注意事项

    前几天遇到了软件模拟spi的时候,读和写不一致的现象,后来仔细研究了一下,其实是时序性问题不对. spi的有四种时序,硬件实现的时候,很简单,初始化后直接调用api即可.但是软件模拟就比较麻烦. 举例 ...

  8. 剑指offer-面试题56_2-数组中唯一只出现一次的数字-位运算

    /* 题目: 数组中除一个数字只出现一次外,其余数字都出现3次. */ /* 思路: 位运算. */ #include<iostream> #include<cstring> ...

  9. npm 安装与部署

    nodejs 安装 查看版本 官方网址 下载linux版,64位 wget https://npm.taobao.org/mirrors/node/v12.13.1/node-v12.13.1-lin ...

  10. 一文看懂AI深度学习丨曼孚科技

    深度学习(Deep Learning)是机器学习的一种,而机器学习是实现人工智能的必经途径. 目前大部分表现优异的AI应用都使用了深度学习技术,引领了第三次人工智能的浪潮. 一. 深度学习的概念 深度 ...