033.搜索旋转排序数组

2024-06-08 02:28
文章标签 数组 搜索 排序 旋转 033

本文主要是介绍033.搜索旋转排序数组,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题意

整数数组 nums 按升序排列,数组中的值 互不相同

在传递给方法之前,nums 在预先未知的某个下标 k(0 <= k < nums.length)上进行了旋转,使数组变为 [nums[k], nums[k+1], ..., nums[n-1], nums[0], nums[1], ..., nums[k-1]](下标 从 0 开始计数)。例如, [0,1,2,4,5,6,7] 在下标 3 处经旋转后可能变为 [4,5,6,7,0,1,2] 。

给你旋转后的数组 nums 和一个整数 target,如果 nums 中存在这个目标值 target,则返回它的下标,否则返回 -1 。

你必须设计一个时间复杂度为 O(log n) 的算法解决此问题。

难度

中等

示例

输入:nums = [4,5,6,7,0,1,2], target = 0
输出:4输入:nums = [4,5,6,7,0,1,2], target = 3
输出:-1

分析 1

这道题咋眼一看,有点绕,给一个旋转后的数组,和一个目标值,并要求我们找到这个目标值的下标。

但稍微分析一下就知道,就是在一个数组当中查找一个目标值,如果不考虑时间复杂度的话,我们可以直接遍历一遍,找到就返回下标,找不到就返回-1。

超级简单:

class Solution {public int search(int[] nums, int target) {for(int i = 0;i < nums.length;i++){if(nums[i] == target)return i;}return -1;}
}

来看一下效率:

效率不错,但要时间复杂度为对数级别。那就要考虑别的算法

分析 2

我们知道,在一个有序的数组里面去判断一个数是否存在,可以利用二分查找,时间复杂度刚好就是$O(\log{n})$。

但这道题只是部分有序(因为旋转了),该怎么去判断呢?

闭上眼,想一下。

从任意位置将这个部分有序的数组分开,那么分开之后的两部分必然有一部分是有序的!

比如:

nums = [4,5,6,7,0,1,2]      
nums = [4] [5,6,7,0,1,2]	breakPos = 1
nums = [4,5] [6,7,0,1,2]    breakPos = 2
nums = [4,5,6] [7,0,1,2]    breakPos = 3
nums = [4,5,6,7] [0,1,2]    breakPos = 4
nums = [4,5,6,7,0] [1,2]    breakPos = 5
nums = [4,5,6,7,0,1] [2]    breakPos = 6

果然,至少有一部分是有序的。那我们是不是就可以从有序的部分当中去寻找 target 呢?

可以是可以,但时间复杂度并不是 $O(\log{n})$,还要加上 breakPos 分割后无序的部分,合起来就是$O(n + \log{n})$,显然也不符合题目的要求。

考虑下面的思路:

  • 如果[lef,breakPos - 1]是有序的,而且nums[lef] <= target && target < nums[breakPos],那么答案肯定在[lef, breakPos - 1],直接调整上界rig到breakPos - 1。
  • 如果[breakPos,rig]是有序的,而且nums[breakPos] < target && target <= nums[rig],那么答案肯定在[breakPos,rig]中,调整下界lef到breakPos即可。

也就是说,我们通过判断有序的部分,来决定下一步的查找范围。

  • 只有在有序区间内才可以通过区间两端的数值判断 target 是否在其中。
  • 判断有序区间还是乱序区间:left <= right 是顺序区间,否则乱序区间。
  • 每次二分至少存在一个有序区间。
class Solution {public int search(int[] nums, int target) {int siz = nums.length; // 数组长度int lef = 0, rig = siz - 1; // 初始化左右指针while (lef <= rig) { // 当左指针小于等于右指针时进行循环int mid = (lef + rig) >> 1; // 计算中间索引,使用右移运算符相当于除以 2if (nums[mid] == target) // 如果中间元素等于目标值,返回中间索引return mid;if (nums[0] <= nums[mid]) { // 如果左半部分有序if (nums[0] <= target && target < nums[mid]) // 如果目标值在左半部分范围内rig = mid - 1; // 移动右指针elselef = mid + 1; // 否则移动左指针} else { // 右半部分有序if (nums[mid] < target && target <= nums[siz - 1]) //如果目标值在右半部分范围内lef = mid + 1; // 移动左指针elserig = mid - 1; // 否则移动右指针}}return -1; // 如果未找到目标值,返回 -1}
}

我们来分析一下代码的关键部分:

第一步,初始化

  • siz:数组长度。
  • lef 和 rig:左右指针,分别初始化为数组的第一个和最后一个索引。

第二步,二分查找

计算中间索引 mid;如果 nums[mid] 等于目标值 target,直接返回 mid。

检查数组的左半部分是否有序(nums[0] <= nums[mid])。

如果左半部分有序,并且目标值在左半部分范围内(nums[0] <= target && target < nums[mid]),移动右指针到 mid - 1;否则,移动左指针到 mid + 1。

如果右半部分有序,并且目标值在右半部分范围内(nums[mid] < target && target <= nums[siz - 1]),移动左指针到 mid + 1;否则,移动右指针到 mid - 1。

第三步,如果循环结束仍未找到目标值,返回 -1。

假设数组为 [4, 5, 6, 7, 0, 1, 2],目标值 target 为 0,我们来模拟一下整个题解过程:

  • 1.初始 lef = 0,rig = 6,mid = 3,nums[mid] = 7。
  • 2.nums[0] <= nums[mid],左半部分有序。
  • 3. 0 不在左半部分范围内,移动左指针 lef = 4。
  • 4.更新 mid = 5,nums[mid] = 1。
  • 5.nums[4] <= nums[mid],右半部分有序。
  • 6. 0 在右半部分范围内,移动右指针 rig = 5。
  • 7.更新 mid = 4,nums[mid] = 0,找到目标值,返回 4。

总结

遇到 O(log n) 的题目,首先想到的就是二分查找,这道题也是一样,只不过在二分查找的基础上,增加了一些判断条件。而二分查找的关键是找到有序的部分。

力扣链接:. - 力扣(LeetCode)

这篇关于033.搜索旋转排序数组的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1040981

相关文章

HTML5 搜索框Search Box详解

《HTML5搜索框SearchBox详解》HTML5的搜索框是一个强大的工具,能够有效提升用户体验,通过结合自动补全功能和适当的样式,可以创建出既美观又实用的搜索界面,这篇文章给大家介绍HTML5... html5 搜索框(Search Box)详解搜索框是一个用于输入查询内容的控件,通常用于网站或应用程

MySQL JSON 查询中的对象与数组技巧及查询示例

《MySQLJSON查询中的对象与数组技巧及查询示例》MySQL中JSON对象和JSON数组查询的详细介绍及带有WHERE条件的查询示例,本文给大家介绍的非常详细,mysqljson查询示例相关知... 目录jsON 对象查询1. JSON_CONTAINS2. JSON_EXTRACT3. JSON_TA

Java List排序实例代码详解

《JavaList排序实例代码详解》:本文主要介绍JavaList排序的相关资料,Java排序方法包括自然排序、自定义排序、Lambda简化及多条件排序,实现灵活且代码简洁,文中通过代码介绍的... 目录一、自然排序二、自定义排序规则三、使用 Lambda 表达式简化 Comparator四、多条件排序五、

JAVA数组中五种常见排序方法整理汇总

《JAVA数组中五种常见排序方法整理汇总》本文给大家分享五种常用的Java数组排序方法整理,每种方法结合示例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录前言:法一:Arrays.sort()法二:冒泡排序法三:选择排序法四:反转排序法五:直接插入排序前言:几种常用的Java数组排序

使用animation.css库快速实现CSS3旋转动画效果

《使用animation.css库快速实现CSS3旋转动画效果》随着Web技术的不断发展,动画效果已经成为了网页设计中不可或缺的一部分,本文将深入探讨animation.css的工作原理,如何使用以及... 目录1. css3动画技术简介2. animation.css库介绍2.1 animation.cs

Java数组初始化的五种方式

《Java数组初始化的五种方式》数组是Java中最基础且常用的数据结构之一,其初始化方式多样且各具特点,本文详细讲解Java数组初始化的五种方式,分析其适用场景、优劣势对比及注意事项,帮助避免常见陷阱... 目录1. 静态初始化:简洁但固定代码示例核心特点适用场景注意事项2. 动态初始化:灵活但需手动管理代

C++中初始化二维数组的几种常见方法

《C++中初始化二维数组的几种常见方法》本文详细介绍了在C++中初始化二维数组的不同方式,包括静态初始化、循环、全部为零、部分初始化、std::array和std::vector,以及std::vec... 目录1. 静态初始化2. 使用循环初始化3. 全部初始化为零4. 部分初始化5. 使用 std::a

shell编程之函数与数组的使用详解

《shell编程之函数与数组的使用详解》:本文主要介绍shell编程之函数与数组的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录shell函数函数的用法俩个数求和系统资源监控并报警函数函数变量的作用范围函数的参数递归函数shell数组获取数组的长度读取某下的

Mybatis 传参与排序模糊查询功能实现

《Mybatis传参与排序模糊查询功能实现》:本文主要介绍Mybatis传参与排序模糊查询功能实现,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录一、#{ }和${ }传参的区别二、排序三、like查询四、数据库连接池五、mysql 开发企业规范一、#{ }和${ }传参的

C++原地删除有序数组重复项的N种方法

《C++原地删除有序数组重复项的N种方法》给定一个排序数组,你需要在原地删除重复出现的元素,使得每个元素只出现一次,返回移除后数组的新长度,不要使用额外的数组空间,你必须在原地修改输入数组并在使用O(... 目录一、问题二、问题分析三、算法实现四、问题变体:最多保留两次五、分析和代码实现5.1、问题分析5.