LeetCode 面试题 10.03. 搜索旋转数组

文章目录

一、题目

搜索旋转数组。给定一个排序后的数组,包含n个整数,但这个数组已被旋转过很多次了,次数不详。请编写代码找出数组中的某个元素,假设数组元素原先是按升序排列的。若有多个相同元素,返回索引值最小的一个。

示例1:

输入: arr = [15, 16, 19, 20, 25, 1, 3, 4, 5, 7, 10, 14], target = 5
输出: 8(元素5在该数组中的索引)

示例2:

输入: arr = [15, 16, 19, 20, 25, 1, 3, 4, 5, 7, 10, 14], target = 11
输出: -1 (没有找到)

提示:

  • arr 长度范围在[1, 1000000]之间

点击此处跳转题目

二、C# 题解

类似二分查找,但由于优先返回第一次出现的 target,因此主要思想如下:

  • 取中间元素 arr[mid],判断是否为 target。如果是,不直接返回,而转到步骤 2。
  • 查找左半部分,结果记为 left。如果查找到结果(left >= 0),则直接返回 left。
  • 若未找到,最后查找右半部分,返回结果。

上述方式类似前序遍历树结构,返回第一个查找的值。

同时,做出如下剪枝优化:

  1. 数组被旋转,则数组内所有元素都在区间 [ a r r [ i ] , + ∞ ) ∪ ( − ∞ , a r r [ j ] ] [arr[i], +\infty)\cup(-\infty,arr[j]] [arr[i],+∞)∪(−∞,arr[j]] 内。若 target 不在该区间内,直接返回 -1。
    15 , 16 , 19 , 20 , 25 , 1 , 3 , 4 , 5 , 7 , 10 t a r g e t : 12 \begin{aligned} & \sout{15, 16, 19, 20, 25, 1, 3, 4, 5, 7, 10} & target:12 \\ \end{aligned} 15,16,19,20,25,1,3,4,5,7,10target:12

  2. 数组未被旋转(有序),则数组内所有元素都在区间 [ a r r [ i ] , a r r [ j ] ] [arr[i], arr[j]] [arr[i],arr[j]] 内。若 target 不在该区间内,直接返回 -1。
    1 , 3 , 4 , 5 , 7 , 10 , 15 , 16 , 19 , 20 , 25 t a r g e t : 30 \begin{aligned} & \sout{1, 3, 4, 5, 7, 10, 15, 16, 19, 20, 25} & target:30 \\ \end{aligned} 1,3,4,5,7,10,15,16,19,20,25target:30

  3. mid 查找到值,即 arr[mid] == target。则只查看左枝,减去右枝。
    { 15 , 16 , 19 , 20 , 1 } , 2 ‾ , 3 , 4 , 5 , 7 , 10 t a r g e t : 2 \begin{aligned} & \{15, 16, 19, 20, 1\}, \underline{\bold{2}}, \sout{3, 4, 5, 7, 10} & target:2 \\ \end{aligned} {15,16,19,20,1},2,3,4,5,7,10target:2

  4. 右枝有序且 target 在该区间内,则忽略左枝,只看右枝。
    15 , 16 , 19 , 20 , 1 , 2 ‾ , { 3 , 4 , 5 , 7 , 10 } t a r g e t : 5 \begin{aligned} & \sout{15, 16, 19, 20, 1}, \underline{2}, \{3, 4, 5, 7, 10\} & target:5 \\ \end{aligned} 15,16,19,20,1,2,{3,4,5,7,10}target:5

csharp 复制代码
public class Solution {
    public int Search(int[] arr, int target) {
        return Partition(arr, 0, arr.Length - 1, target);
    }

    public int Partition(int[] arr, int i, int j, int target) {
        if (i > j) return -1; // 递归出口

        // 剪枝
        if (arr[j] < target && target < arr[i]) return -1; // case 1, 直接返回
        if (target < arr[i] && target > arr[j]) return -1; // case 2, 直接返回

        int mid = (i + j) / 2, left;
        if (arr[mid] == target) { // case 3, 减去右半部分
            left = Math.Min(Partition(arr, i, mid - 1, target), mid);
            return left == -1 ? mid : left;
        }
        if (arr[mid] < target && target < arr[j]) // case 4, 减去左半部分
            return Partition(arr, mid + 1, j, target);

        // 优先返回最前面的结果
        left = Partition(arr, i, mid - 1, target);
        if (left != -1) return left;
        return Partition(arr, mid + 1, j, target);
    }
}
  • 时间:84 ms,击败 100.00% 使用 C# 的用户
  • 内存:39.92 MB,击败 100.00% 使用 C# 的用户

还可以先用二分查找转折点 k,此时考虑起点为 k 终点为 k - 1 的循环数组,即,从 [k, j] 续上 [i, k - 1] 的有序数组,对其使用二分查找第一个元素。详细代码这里就不附上了。

↱ k 15 , 16 , 19 , 20 , 1 ‾ , 2 , 3 , 4 , 5 , 7 , 10 t a r g e t : 5 ⇓ − , − , − , − , 1 ‾ , 2 , 3 , 4 , 5 , 7 , 10 , 15 , 16 , 19 , 20 t a r g e t : 5 \begin{aligned} & \hspace{5.7em} \Rsh k \\ & 15, 16, 19, 20, \underline{1}, 2, 3, 4, 5, 7, 10 & target:5 \\ & \hspace{5.7em} \Downarrow\\ & -,\ -,\ -,\ -, \hspace{0.1em} \underline{1}, 2, 3, 4, 5, 7, 10, 15, 16, 19, 20 & target:5 \\ \end{aligned} ↱k15,16,19,20,1,2,3,4,5,7,10⇓−, −, −, −,1,2,3,4,5,7,10,15,16,19,20target:5target:5

相关推荐
冠位观测者2 分钟前
【Leetcode 每日一题】2545. 根据第 K 场考试的分数排序
数据结构·算法·leetcode
@小码农16 分钟前
202411 第十六届蓝桥杯青少组 STEMA 考试真题 汇总
职场和发展·蓝桥杯
向宇it17 分钟前
【从零开始入门unity游戏开发之——unity篇02】unity6基础入门——软件下载安装、Unity Hub配置、安装unity编辑器、许可证管理
开发语言·unity·c#·编辑器·游戏引擎
古希腊掌管学习的神1 小时前
[搜广推]王树森推荐系统笔记——曝光过滤 & Bloom Filter
算法·推荐算法
qystca1 小时前
洛谷 P1706 全排列问题 C语言
算法
yngsqq1 小时前
一键打断线(根据相交点打断)——CAD c# 二次开发
windows·microsoft·c#
古希腊掌管学习的神1 小时前
[LeetCode-Python版]相向双指针——611. 有效三角形的个数
开发语言·python·leetcode
浊酒南街1 小时前
决策树(理论知识1)
算法·决策树·机器学习
就爱学编程1 小时前
重生之我在异世界学编程之C语言小项目:通讯录
c语言·开发语言·数据结构·算法
ProcessOn官方账号1 小时前
如何绘制网络拓扑图?附详细分类解说和用户案例!
网络·职场和发展·流程图·拓扑学