LeetCode Hot100(50/100)——153. 寻找旋转排序数组中的最小值

一、题目简介

题目来源:LeetCode 官方题目描述

给定一个 升序排列数组,该数组在某个未知的轴上进行了旋转,要求找出其中的最小元素。

例如:

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

数组原本是 [1, 2, 3, 4, 5],在元素 3 处进行了旋转,最小值为 1。


二、题意分析

这个问题的本质是:
在一个被旋转过的有序数组中快速找到最小值。
由于原数组是递增的,只是被某个点切开后拼接,所以仍保留二分查找的性质。


三、思维导图

寻找旋转排序数组中的最小值

题目理解

升序数组旋转

找最小元素

解法思路

二分查找

比较 mid 与右端点 数值关系

缩小搜索区间

特殊情况

未旋转的纯升序数组

数组中只有一个元素

复杂度分析

时间 O(log n)

空间 O(1)

Java实现

while循环

条件判断与区间更新


四、算法思路详解

思路一:线性扫描(暴力解法)

最直接的方式是遍历整个数组,找到最小值:

for i in nums:
    minValue = min(minValue, i)
  • 时间复杂度O(n)
  • 空间复杂度O(1)
  • 缺点:无法利用数组的有序性,效率较低。

思路二:二分查找优化解法 —— 核心算法

由于旋转后的数组依然保持局部有序结构,我们可以通过 二分查找 实现高效查找。

核心逻辑
  1. 初始化左右指针:left = 0, right = nums.length - 1
  2. left < right 时:
    • 计算中点 mid = (left + right) / 2
    • 对比 nums[mid]nums[right]
  3. 根据比较结果更新搜索区间:
    • nums[mid] > nums[right]:最小值在右侧区间 → left = mid + 1
    • nums[mid] <= nums[right]:最小值在左侧区间或 mid → right = mid
  4. 最终 leftright 相遇时,即最小值索引。

五、流程图示意

开始

初始化 left=0, right=n-1

left < right ?

返回 nums[left]

计算 mid = (left + right)/2

nums[mid] > nums[right]?

left = mid + 1

right = mid

结束


六、Java代码实现

下面是基于上述思路的 Java 示例实现:

public class Solution {
    public int findMin(int[] nums) {
        int left = 0;
        int right = nums.length - 1;
        while (left < right) {
            int mid = left + (right - left) / 2;
            if (nums[mid] > nums[right]) {
                // 最小值在右侧区间
                left = mid + 1;
            } else {
                // 最小值在左侧区间或 mid 位置
                right = mid;
            }
        }
        // left == right 时即为最小值
        return nums[left];
    }
}

七、复杂度分析

指标复杂度说明
时间复杂度O(log n)使用二分查找,每次缩小一半搜索区间
空间复杂度O(1)只使用常量级变量存储指针与中间值

八、示例验证

输入:

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

查找过程如下:

步骤leftrightmidnums[mid]nums[right]区间更新
106372nums[mid] > nums[right] → left = 4
246512nums[mid] <= nums[right] → right = 5
345401nums[mid] <= nums[right] → right = 4

此时 left == right == 4,最小值为 nums[4] = 0


九、总结

解法思路时间复杂度空间复杂度
暴力扫描遍历整个数组寻找最小值O(n)O(1)
二分查找利用有序性缩小搜索区间O(log n)O(1)

最终采取 二分查找解法 可在对数时间内高效求解。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

TracyCoder123

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

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

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

打赏作者

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

抵扣说明:

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

余额充值