壹 ❀ 引

堕落了一天,那么接着来刷leetcode,今天做的一题不算复杂,题目来自leetcode153. 寻找旋转排序数组中的最小值,题目描述如下:

假设按照升序排序的数组在预先未知的某个点上进行了旋转。

( 例如,数组 [0,1,2,4,5,6,7] 可能变为 [4,5,6,7,0,1,2] )。

请找出其中最小的元素。

你可以假设数组中不存在重复元素。

示例 1:

输入: [3,4,5,1,2]
输出: 1

示例 2:

输入: [4,5,6,7,0,1,2]
输出: 0

在之前JS leetcode 旋转数组 题解分析一文中,我们已经做过旋转数组的题目,所以这里所说的旋转其实只是将数组尾部的元素依次加入数组头部的操作。我们简单分析题目,再说怎么实现。

贰 ❀ 解题思路

贰 ❀ 暴力解题

由题目提供的信息可知,数组为升序排列数组,而且在某个未知的点进行了旋转,所以它可能没转。

不过但站在找出数组中最小元素来说,我们可以不考虑这些条件,直接祭出Math.min()大法:

/**
* @param {number[]} nums
* @return {number}
*/
var findMin = function(nums) {
return Math.min(...nums); //ES5
//return Math.min.apply(null, nums);
};

由于数组原本就是排序好的,只是可能进行了旋转,我们可以可耻的将数组再排序然后取出索引0位元素:

/**
* @param {number[]} nums
* @return {number}
*/
var findMin = function (nums) {
return nums.sort((a, b) => a - b)[0];
};

虽然能达到目的,总觉得差了点意思,提交后发现执行时间排名也很低,说明有更好的做法。

贰 ❀ 什么是二分法?

在看了官方的解题思路后,发现本题用二分法查找会更好,不过对于我来说,二分法是啥都比较疑惑,所以这里先简单说说什么是二分法。

比如,我们要在数组[1,2,3,4,5,6,7,8,9]里面找到目标9的下标,按照常规遍历,我们只有从头遍历到最后一位才能找到,时间复杂度为O(n),使用二分法就不一样,如下:

/**
* @desc 二分法查找目标元素索引
* @param {*} arr 数组
* @param {*} target 目标元素
*/
function binarySearch(arr, target) {
// 数组起始索引
var low = 0;
// 数组最后一项索引
var high = arr.length - 1;
while (low <= high) {
// 获取数组中间项索引
var mid = Math.floor((low + high) / 2);
if (target === arr[mid]) {
// 返回目标元素下标
return mid;
// 根据比较来决定下次查找的数组应该是左还是右
} else if (target > arr[mid]) {
low = mid + 1;
} else {
high = mid - 1;
};
};
// 没找到返回-1
return -1;
};
binarySearch([1, 2, 3, 4, 5, 6, 7, 8, 9], 9); //8

我们一开始就可以按照某个特定规则找到数组中的中间项元素,从而将数组一分为二,然后将中间项与目标元素比较,比如一开始我们找到了5,由于比目标元素9小,所以9肯定在5右边的数组中。

那么下次遍历就从[6,7,8,9]开始,这次中间元素找到7,仍然比9小,继续上述操作,又在[8,9]中寻找。直到最后找到目标元素9,从而获取到索引。

你看,这样对半分的查找,是不是比我们常规从头到尾的遍历要快很多,二分法的时间复杂度为O(logn),注意,二分法适合有序序列,不然我们也不知道下次应该去左边还是右边比较了。

贰 ❀ 通过二分法解决此题

前面说了,二分法适合有序数组,这样我们才好根据特定条件来决定下次应该从哪边开始,比如上文中的target > arr[mid]。而本题虽然也是有序,但很遗憾的是数组经过了旋转操作,所以一般的二分法并不适用。

比较麻烦的是,虽然题目说了做了旋转,不保证数组旋转了一整圈结果并无变化的情况。所以第一步我们可以先判断数组到底有没有被旋转,判断条件很简单

已知数组是有序数组,如果数组发生了旋转,那么数组第一位一定大于最后一位,反之如果数组未发生旋转,那么第一位一定小于最后一位,这种情况我们直接返回第一位即可。

那如果数组已发生了旋转,我们又该怎么判断呢,其实有一个这样的特点:

纵观整个数组,我们以相邻两个元素来看,假设当前中间元素为mid,如果arr[mid-1]大于arr[mid],那么mid为我们想要的元素。或者arr[mid]大于arr[mid+1],那么此时mid+1是我们想要找的元素。为啥这么说,因为数组虽然旋转了,但相邻且满足如上任意条件之一的情况只存在一次,不信大家随便写个有序数组看。

所以每次确定中间元素,我们都得走如下条件,只要满足其一即是我们想要找的元素。

if (nums[mid] > nums[mid + 1]) {
return nums[mid + 1];
}; if (nums[mid - 1] > nums[mid]) {
return nums[mid];
};

这是找到目标元素的条件,那不满足我们如何知道应该查找左边数组还是右边数组呢?其实有这样一个规律:

如上图,我们将9=>1之间称为变化点,变化点左边的元素都一定比数组第一位元素大,变化点右边的所有元素,一定比数组第一位元素小。

所以找到中间元素mid,如果mid比第一位还大,那说明我们应该去数组右边找,如果mid比第一位还小,那么应该去左边找。

你可能在想,万一我这个mid第一次就是最小元素咋办,如果运气真这么好,它早就被我们上面定的两个条件之一给返回了,能走到这一步说明这个mid一定不是我们想找的目标元素,这才要分去哪边找啊。

那么我们上代码:

/**
* @param {number[]} nums
* @return {number}
*/
var findMin = function(nums) {
// 假设数组只有一项,直接返回
if (nums.length == 1) {
return nums[0];
}; var left = 0,
right = nums.length - 1;
// 假设数组最后一项大于第一项,说明数组未旋转,直接返回
if (nums[right] > nums[0]) {
return nums[0];
}; // 既然能走到这一步,那说明数组一定旋转了,套用之前的规则,使用二分法进行查找
while (right >= left) {
// Find the mid element
var mid = Math.floor((left + right) / 2);
// 满足如下条件之一说明就是最小元素,直接返回即可
if (nums[mid] > nums[mid + 1]) {
return nums[mid + 1];
};
if (nums[mid - 1] > nums[mid]) {
return nums[mid];
};
// 比较当前中间元素与第一位
if (nums[mid] > nums[0]) {
// 如果要大,那就去右边找
left = mid + 1;
} else {
// 反之就去左边找
right = mid - 1;
};
};
return -1;
};

执行图解如下:

其实这段代码我分析了很久,对于我觉得比较难的是什么情况返回mid,我觉得mid如果是最小,它一定比mid+1小,但这样是不成立的,例如1比2小,2也比3小,这个条件没法用。

所以官方分析我觉得让我佩服的是,整个数组中,当mid是1时,mid-1一定比mid大,整个数组你找不出第二个这样的情况。同理,当mid是9时,mid一定比mid+1大,你同样找不出第二个这样的情况...

所以变化点相关的两个元素9和1才是解题关键。

那么关于本题就分析到这里了。

相关题型:

JS Leetcode 154. 寻找旋转排序数组中的最小值 II 题解分析

JS Leetcode 33. 搜索旋转排序数组题解,图解旋转数组中的二分法

JS Leetcode 81. 搜索旋转排序数组 II 题解,补救二分法的可行性

另外,二分法参考如下:

算法——二分法查找(binarySearch)

JS leetcode 寻找旋转排序数组中的最小值 题解分析,你不得不了解的二分法的更多相关文章

  1. LeetCode:寻找旋转排序数组中的最小值【153】

    LeetCode:寻找旋转排序数组中的最小值[153] 题目描述 假设按照升序排序的数组在预先未知的某个点上进行了旋转. ( 例如,数组 [0,1,2,4,5,6,7] 可能变为 [4,5,6,7,0 ...

  2. Leetcode之二分法专题-154. 寻找旋转排序数组中的最小值 II(Find Minimum in Rotated Sorted Array II)

    Leetcode之二分法专题-154. 寻找旋转排序数组中的最小值 II(Find Minimum in Rotated Sorted Array II) 假设按照升序排序的数组在预先未知的某个点上进 ...

  3. Leetcode之二分法专题-153. 寻找旋转排序数组中的最小值(Find Minimum in Rotated Sorted Array)

    Leetcode之二分法专题-153. 寻找旋转排序数组中的最小值(Find Minimum in Rotated Sorted Array) 假设按照升序排序的数组在预先未知的某个点上进行了旋转. ...

  4. [LeetCode每日一题]153.寻找旋转排序数组中的最小值

    [LeetCode每日一题]153.寻找旋转排序数组中的最小值 问题 已知一个长度为 n 的数组,预先按照升序排列,经由 1 到 n 次 旋转 后,得到输入数组.例如,原数组 nums = [0,1, ...

  5. 【leetcode】153. 寻找旋转排序数组中的最小值

    题目链接:传送门 题目描述 现有一个有序数组,假设从某个数开始将它后面的数按顺序放到了数组前面.(即 [0,1,2,4,5,6,7] 可能变成 [4,5,6,7,0,1,2]). 请找出数组中的最小元 ...

  6. [LeetCode] 154. 寻找旋转排序数组中的最小值 II

    题目链接 : https://leetcode-cn.com/problems/find-minimum-in-rotated-sorted-array-ii/ 题目描述: 假设按照升序排序的数组在预 ...

  7. Java实现 LeetCode 154 寻找旋转排序数组中的最小值 II(二)

    154. 寻找旋转排序数组中的最小值 II 假设按照升序排序的数组在预先未知的某个点上进行了旋转. ( 例如,数组 [0,1,2,4,5,6,7] 可能变为 [4,5,6,7,0,1,2] ). 请找 ...

  8. Java实现 LeetCode 153 寻找旋转排序数组中的最小值

    153. 寻找旋转排序数组中的最小值 假设按照升序排序的数组在预先未知的某个点上进行了旋转. ( 例如,数组 [0,1,2,4,5,6,7] 可能变为 [4,5,6,7,0,1,2] ). 请找出其中 ...

  9. lintcode: 寻找旋转排序数组中的最小值

    寻找旋转排序数组中的最小值 假设一个旋转排序的数组其起始位置是未知的(比如0 1 2 4 5 6 7 可能变成是4 5 6 7 0 1 2). 你需要找到其中最小的元素. 你可以假设数组中不存在重复的 ...

  10. [Swift]LeetCode154. 寻找旋转排序数组中的最小值 II | Find Minimum in Rotated Sorted Array II

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

随机推荐

  1. Guava缓存工具类封装和使用

    本文为博主原创,未经允许不得转载: Guava是谷歌提供的一款强大的java工具库,里面包含了很多方便且高效的工具,在项目开发中有业务场景需要保存数据到内存当中, 且只需要保存固定时间就可以,该数据只 ...

  2. 2. 成功使用SQL Plus完成连接,但在使用Oracle SQL Developer连接时,发生报错ORA-12526: TNS:listener: all appropriate instances are in restricted mode

    经了解后得知,错误原因:ORA-12526: TNS: 监听程序: 所有适用例程都处于受限模式. 解决办法:使用系统管理员身份运行以下一段代码 ALTER SYSTEM DISABLE RESTRIC ...

  3. Qt5.9 UI设计(三)——添加UI、类及资源文件

    前言 设计一个软件,最简单的方式就是把控件直接往UI上放,然后再把功能实现了.这样可以实现基本的功能,但是界面不能缩放,如果拖动软件改变界面的大小,界面上的控件就会乱成一团,或者是界面的控件压根就不会 ...

  4. [转帖]故障分析 | 让top命令直接显示Java线程名?-- 解析OpenJDK的一个bug修复

    https://zhuanlan.zhihu.com/p/413136873 作者:阎虎青DBLE 开源项目负责人,负责分布式数据库中间件研发工作:持续专注于数据库方面的技术,始终在一线从事开发:对数 ...

  5. [转帖]datax安装+配置+使用文档

    1 DataX离线同步工具DataX3.0介绍 DataX 是阿里巴巴集团内被广泛使用的离线数据同步工具/平台,实现包括 MySQL.Oracle.SqlServer.Postgre.HDFS.Hiv ...

  6. [转帖]TiDB 数据库核心原理与架构 [TiDB v6](101)笔记

    https://www.jianshu.com/p/01e49a93f671 description: "本课程专为将在工作中使用 TiDB 数据库的开发人员.DBA 和架构师设计. 本门课 ...

  7. [转帖]shell脚本之awk命令——按列求平均值、最大值、最小值

    文章目录 写在前面 awk求平均值 awk求最大值 awk求最小值 awk求极值.均值的实际应用 写在前面 awk命令求极值和均值需要熟悉该命令的基本用法,如果你不熟悉该命令,请先阅读shell脚本之 ...

  8. 使用systemd管理多nginx服务以及单nginx服务实现多vhost访问的操作步骤

    背景 nginx是开源的web服务器, 性能与可配置性和插件做的非常完善. 可以使用简单的命令拉起来nginx进行服务提供,但是有时候需要使用keepalive等软件实现保活,以及实现开启启动等,比较 ...

  9. k8s的内部服务通信

    首先看看 k8s 集群中内部各个服务互相访问的方法 Cluster IP Kubernetes以Pod作为应用部署的最小单位.Kubernetes会根据Pod的声明对其进行调度,包括创建.销毁.迁移. ...

  10. 【记录一个问题】vm-select和vm-storage均无法做并行查询

    作者:张富春(ahfuzhang),转载时请注明作者和引用链接,谢谢! cnblogs博客 zhihu Github 公众号:一本正经的瞎扯 看我提的这个issue: need parallel qu ...