题目描述
把一个数组最开始的若干个元素搬到数组的末尾,我们称之为数组的旋转。 输入一个非减排序的数组的一个旋转,输出旋转数组的最小元素。 例如数组{3,4,5,1,2}为{1,2,3,4,5}的一个旋转,该数组的最小值为1。 NOTE:给出的所有元素都大于0,若数组大小为0,请返回0。
成都网站建设哪家好,找创新互联!专注于网页设计、重庆网站建设、微信开发、小程序开发、集团成都企业网站定制等服务项目。核心团队均拥有互联网行业多年经验,服务众多知名企业客户;涵盖的客户类型包括:花箱等众多领域,积累了大量丰富的经验,同时也获得了客户的一致赞美!
class Solution:
"""
由于整个数组在一定程度上是有序的,因此可以借鉴二分查找的思想,达到接近O(logn)的时间复杂度。
将一个有序(升序)数组的前x个元素挪到末尾,这里可以分类进行讨论。
第一类:挪动元素个数为数组长度的整数倍,那么这时候等于没有挪动,最小值出现在idx=0
第二类:挪动元素个数不是数组长度的整数倍,那么这时候挪动后的数组可以分成两个子数组,其中左边子
数组的元素都是大于等于右边子数组的。
[3, 4, 5, 1, 2]
这时候我们维护两个指针p1和p2,分别指向左边子数组和右边子数组。当中间元素大于等于p1指向
的元素的时候,则中间元素属于左边子数组,反之属于右边子数组。
当两个指针相邻的时候p2就指向了右边子数组的第一个元素,也就是整个数组的最小值。
第三类:[1, 0, 1, 1, 1]
当p1和p2指向的元素和中间的元素相等的时候,这时候如果按照第二类的思路,那么会误判最小值
在中间元素之后,因此这种情况下我们需要顺序查找。
"""
def minNumberInRotateArray(self, rotateArray):
if not rotateArray:
return 0
# 将mid初始化为0,可以处理第一类情况,因为这时不会进入循环,直接输出最小值
left, mid, right = 0, 0, len(rotateArray) - 1
while rotateArray[left] >= rotateArray[right]:
# 如果left和right已经相邻,那么最小值就是right指向的元素
if left == right - 1:
mid = right
break
mid = (left + right) >> 1
# 如果left, right, mid指向的元素都相等,那么需要对这个区间进行顺序查找,否则按照
# 第二类情况的解题思路会判断错误
if rotateArray[left] == rotateArray[mid] == rotateArray[right]:
return min(rotateArray[left:right + 1])
# 中间元素在左半边,最小值出现在右边子数组[mid, right]
if rotateArray[left] <= rotateArray[mid]:
left = mid
# 中间元素在右半边,最小值出现在左边子数组[left, mid]
else:
right = mid
return rotateArray[mid]
网站栏目:剑指offer:旋转数组的最小值
当前链接:https://www.cdcxhl.com/article32/gjospc.html
成都网站建设公司_创新互联,为您提供网站建设、网站排名、软件开发、ChatGPT、Google、静态网站
声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 创新互联