搜索旋转数组

双指针 同时被 2 个专栏收录
17 篇文章 0 订阅
86 篇文章 0 订阅

面试题 10.03. 搜索旋转数组

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

题目描述

面试题 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]之间

思路分析

打眼一瞅直接暴力

代码

class Solution {
public:
    int search(vector<int>& arr, int target) {
        // vilonce
        for(int i=0; i<arr.size(); ++i){
            if(arr[i] == target) return i; 
        }
        return -1;
    }
};

算法分析

- 时间复杂度:O(n)
- 空间复杂度:O(1)

代码改进

说实话感觉这题真的很蠢,为出而出
可以用二分法搜索判断旋转点,用二分法结合之前确定的nums[mid]与旋转点的关系确定如何移动,如果不能确定那就分治解决
关键是要细节考虑很多情况最后实际运行时间并不一定降低多少(小数据)
class Solution {
public:
    int search(vector<int>& arr, int target) {
        int left = 0, right = arr.size() - 1;
    while (left < right) {
        int mid = left + (right - left) / 2;
        if (arr[mid] > arr[left]) {
            if (arr[left] <= target && target <= arr[mid]) {
                right = mid;
            } else {
                left = mid + 1;
            }
        } else if(arr[mid] < arr[left]){
            if (arr[left] <= target || target <= arr[mid]) {     // 如果目标在左边,右边界移动到mid
                right = mid;
            } else {                                               // 否则目标在右半边,左边界移动到mid+1
                left = mid + 1;
            }
        } else {
            if (arr[left] == target) {
                return left;
            } else {
                left ++;
            }
        }
    }
    return arr[left] == target ? left : -1;
    }
};
**********************************************************

相似扩展

  • 相似题目
  • 升级版本

归纳总结

  • 0
    点赞
  • 0
    评论
  • 0
    收藏
  • 一键三连
    一键三连
  • 扫一扫,分享海报

相关推荐
©️2020 CSDN 皮肤主题: 大白 设计师:CSDN官方博客 返回首页
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值