剑指offer 11. 旋转数组的最小数字(很详细!)

本文介绍了如何利用二分查找算法解决旋转数组中找到最小值的题目。通过分析旋转数组的特性,确定最小值可能位于中间位置,并根据元素间的大小关系缩小查找范围。在代码实现中,通过不断调整左右边界,最终找到最小值。时间复杂度为O(logn),空间复杂度为O(1)。
摘要由CSDN通过智能技术生成

剑指offer 11. 旋转数组的最小数字

题目

题目详细

解题思路

一开始,我们就能直接想到,数组找最小值,那么不轻轻松松直接遍历一遍,用一个变量记录最小值,然后直接返回不就完事了?
但是这样做真的不优雅!!!,因为题目已经说了是一个旋转的数组,直接遍历暴力,那么题目还要说旋转干嘛呢?
所以这里我们需要具体分析一下可以包含重复元素的数组旋转过后会是什么情况
旋转过后基本上是这种图案:最小值大概位于中间位置(这里偷点懒直接用了力扣官方的图片,懒得画图了😁)
在这里插入图片描述
注意到,旋转数组中的最后一个值x, 在最小值右侧的值都小于等于x,在最小值左侧的值都大于等于x,因此根据这一条性质,我们可以使用二分查找找出数组中的最小值

我们将在二分查找的每一步中,左边界为 low,右边界为high,区间的中点为 pivot,最小值就在该区间内。

我们将中轴元素 numbers[pivot]与右边界元素 numbers[high] 进行比较,可能会有以下的三种情况:

  • 1.第一种情况是numbers[pivot] < numbers[high] 那么就是这样的
    第一种情况
    说明最小值位于pivot的左边那么我们可以将右边界high缩小为pivot所指的地方high = pivot

  • 第二种情况 numbers[pivot] > numbers[high] ,那么就是这样
    在这里插入图片描述
    说明numbers[pivot]是最小值左侧的元素我们将low更新为 pivot + 1
    low = povit + 1
    这里说明一下为什么是更新为povit + 1 而上面的情况是更新为povit
    因为numbers[pivot] > numbers[high]时,pivot只可能在最小值左侧,pivot+1最大只能是最小值的位置。但是,numbers[pivot] < numbers[high]时,pivot既可能刚好是最小值,也可能在最小值右侧,如果刚好是最小值的话,pivot-1就错过了最小值.

  • 第三种情况是 numbers[pivot] == numbers[high] :
    由于数组中可能存在重复的元素那么可能的情况是这样的
    第三章情况
    由于数组中可能存在重复的元素,我们无法确定pivot是位于最小值的左侧还是右侧.所以我们不能莽撞的去忽略任何一部分的值.但是我们唯一可以确认的是
    不管high指向的值是不是最小值,我们都有一个替代值就是pivot.
    所以我们可以忽略右端点
    即 : high--
    知道了思路我们就可以写代码了

代码

func minArray(numbers []int) int {
	low := 0
	high := len(numbers) - 1
	for low < high {
		povit := low + (high - low)/2
		if numbers[pivot] < numbers[high] {
			high = pivot
		}else if numbers[pivot] > numbers[high]{
			low = pivot + 1
		}else {
			high--
		}
	}
	return numbers[low]
}

时间复杂度 : O(logn) n为numbers的长度,特殊案例时[1,1,1,1]会退化为O(n)
要遍历n次
空间复杂度 : O(1) : low,high,pivot变量使用常数大小的额外空间。
这里使用的是go语言写的,其他语言也差不多啦,解题思路都是一样的

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

gopher333

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值