Leetcode Week4 Find Minimum in Rotated Sorted Array II
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的更多相关文章
- [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 ...
- 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 ...
- 【leetcode】Find Minimum in Rotated Sorted Array II JAVA实现
一.题目描述 Follow up for "Find Minimum in Rotated Sorted Array":What if duplicates are allowed ...
- [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. ...
- 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 ...
- [LeetCode#154]Find Minimum in Rotated Sorted Array II
The question: Follow up for "Find Minimum in Rotated Sorted Array":What if duplicates are ...
- leetcode 【 Find Minimum in Rotated Sorted Array II 】python 实现
题目: Follow up for "Find Minimum in Rotated Sorted Array":What if duplicates are allowed? W ...
- 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. ( ...
- 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. ( ...
随机推荐
- 利用预测分析改进欠款催收策略,控制欺诈风险和信贷风险—— Altair Knowledge Studio 预测分析和机器学习
前提摘要 在数字经济新时代,金融服务主管正在寻求方法去细分他们的产品和市场,保持与客户的联系,寻找能够推动增长和收入的新市场,并利用可以增加优势和降低风险的新技术. 在拥有了众多可用数据之后,金融机构 ...
- HTML连载71-翻转菜单练习
一.翻转菜单练习 <!DOCTYPE html> <html lang="en"> <head> <meta charset=" ...
- linux中vim常用操作
三种模式 # 命令模式 vim 文件名 # 插入模式 按a/i/o 进行插入模式 按esc 重新进入命令模式 # 编辑模式 按:(冒号)进入编辑模式 插入命令 命令 作用 a 在光标所在字符后插入 A ...
- Eversipn STT-MRAM的MJT细胞
业界一直在寻求取代SRAM.其中之一包括自旋转移力矩MRAM(STT-MRAM).新的存储器带来了一些大胆的主张.例如STT-MRAM具有SRAM的速度和闪存的无波动性,具有无限的耐用性. 图1.ST ...
- SDI011 读卡器自动发送00A4选择指令 解决方法
如标题,SDI读卡器会自动发送 004A的应用选择指令 解决方法: 是Certificate Propagation 服务 弄的, 关闭就好了
- Jenkins集成jacoco收集集成测试覆盖率
Jenkins集成jacoco收集集成测试覆盖率 2020-02-28 目录 0 整体思路1 安装版本2 全局工具配置3 Jenkins创建JacocoIntegrateTestDemo项目 3.1 ...
- c#中用office组件读取excel时提示异常来自 HRESULT:0x80010105 (RPC_E_SERVERFAULT)
在excel2007,找到“excel选项”,点开后点击“加载项”,最下面有个管理加载项的下拉菜单,选“COM加载项”,点“转到”,这时会弹出一个框,把里面pdf软件的加载项前面的勾去掉,点确定就ok ...
- Auto-scaling scikit-learn with Apache Spark
来源:https://databricks.com/blog/2016/02/08/auto-scaling-scikit-learn-with-apache-spark.html Data scie ...
- Wannafly Winter Camp 2020 Day 6G 单调栈 - 贪心
对于排列 \(p\),它的单调栈 \(f\) 定义为,\(f_i\) 是以 \(p_i\) 结尾的最长上升子序列的长度 先给定 \(f\) 中一些位置的值,求字典序最小的 \(p\) 使得它满足这些值 ...
- PhpStorm+Xdebug配置单步调试PHP
(一)php安装xdebug扩展,PHPStorm+XDebug单步调试 (二)PHPStorm配置XDebug (三)PHPStorm使用XDebug调试 (四)PhpStorm+Xdebug配置单 ...