33. 搜索旋转排序数组

目录

一、问题描述

二、解题思路

三、代码

四、复杂度分析


一、问题描述

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

在传递给函数之前,nums 在预先未知的某个下标 k0 <= 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) 的算法解决此问题。

二、解题思路

我们需要通过修改标准二分查找的算法来解决旋转数组的问题。具体步骤如下:

  • 取数组的中点 mid,将数组分为左右两部分。因为数组是旋转的,所以至少一部分(左半部分或右半部分)是有序的。
  • 检查目标值 target 是否在有序部分。如果是,则可以在这部分继续进行二分查找;如果不在,则去另一部分查找。
  • 继续重复二分查找,直到找到目标值或确定目标值不存在。

详细步骤:

  • 初始化 leftright 指针分别指向数组的左右两端。
  • left <= right 的条件下进行循环:
    • 计算中点 midmid = left + (right - left) / 2
    • 如果 nums[mid] == target,直接返回 mid
    • 判断哪一半是有序的:
      • 如果 nums[left] <= nums[mid],说明左半部分是有序的。
        • 判断 target 是否在左半部分:如果 nums[left] <= target < nums[mid],则在左半部分查找,更新 right = mid - 1;否则去右半部分查找,更新 left = mid + 1
      • 如果 nums[mid] <= nums[right],说明右半部分是有序的。
        • 判断 target 是否在右半部分:如果 nums[mid] < target <= nums[right],则在右半部分查找,更新 left = mid + 1;否则去左半部分查找,更新 right = mid - 1
  • 如果退出循环时还没有找到目标值,返回 -1。

三、代码

java 复制代码
class Solution {
    public int search(int[] nums, int target) {
        int left = 0, right = nums.length - 1;

        while (left <= right) {
            int mid = left + (right - left) / 2;

            // 如果找到了目标值,返回下标
            if (nums[mid] == target) {
                return mid;
            }

            // 判断左半部分是否有序
            if (nums[left] <= nums[mid]) {
                // 如果目标值在左半部分,则继续在左半部分查找
                if (nums[left] <= target && target < nums[mid]) {
                    right = mid - 1;
                } else {
                    // 否则在右半部分查找
                    left = mid + 1;
                }
            } else {
                // 如果右半部分有序
                if (nums[mid] < target && target <= nums[right]) {
                    left = mid + 1;
                } else {
                    right = mid - 1;
                }
            }
        }

        // 如果没有找到目标值,返回 -1
        return -1;
    }
}

四、复杂度分析

  • 时间复杂度:由于每次将搜索空间缩小一半,因此时间复杂度是 O(log⁡n)O(\log n)O(logn),符合题目要求。
  • 空间复杂度:我们使用的是常数级别的额外空间,空间复杂度为 O(1)O(1)O(1)。
相关推荐
爱喝水的鱼丶35 分钟前
SAP-ABAP:条件判断与循环控制语句(7篇)第七篇:性能优化:条件与循环代码的常见性能瓶颈与优化方案
学习·算法·性能优化·sap·abap
吃好睡好便好5 小时前
提取矩阵某一行或某一列元素
开发语言·人工智能·线性代数·算法·matlab·矩阵
better_liang8 小时前
每日Java面试场景题知识点之-消息队列MQ核心场景与实战
java·面试·kafka·消息队列·rabbitmq·rocketmq·mq
小江的记录本8 小时前
【JVM虚拟机】垃圾回收GC:四种引用类型:强引用、软引用、弱引用、虚引用(附《思维导图》+《面试高频考点清单》)
java·jvm·spring boot·后端·python·spring·面试
圣保罗的大教堂8 小时前
leetcode 2540. 最小公共值 简单
leetcode
小马爱打代码8 小时前
Spring源码 第四篇:Spring 5 源码深度拆解:AOP 全流程核心原理
java·后端·spring
better_liang9 小时前
每日Java面试场景题知识点之-SpringBoot启动流程
java·面试·springboot·源码解析·启动流程
RyFit9 小时前
Java + AI 实战:Spring AI 从入门到企业级落地
java·人工智能·spring
云泽8089 小时前
笔试算法 -位运算篇(二):从唯一字符到消失数字
c++·算法·位运算
ʚ希希ɞ ྀ9 小时前
不同路径|| -- dp
算法