zoukankan      html  css  js  c++  java
  • 【剑指Offer】面试题11. 旋转数组的最小数字

    题目

    把一个数组最开始的若干个元素搬到数组的末尾,我们称之为数组的旋转。输入一个递增排序的数组的一个旋转,输出旋转数组的最小元素。例如,数组 [3,4,5,1,2] 为 [1,2,3,4,5] 的一个旋转,该数组的最小值为1。

    示例 1:

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

    示例 2:

    输入:[2,2,2,0,1]
    输出:0
    

    思路一:暴力

    代码

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

    class Solution {
    public:
        int minArray(vector<int>& numbers) {
            int res = INT_MAX;
            for (int n : numbers) {
                res = min(res, n);
            }
            return res;
        }
    };
    

    思路二:二分

    部分有序可以考虑使用二分,接下来考虑如何通过二分减小范围,本题中左右部分分别是递增有序的,因此最小数可能在 numbers[mid] 左边也可能在右边,如:[1,2,3,4,5]最小数在左边,而[3,4,5,1,2]最小数在右边,所以通过 numbers[mid] 和 numbers[left] 比较无法减少范围。

    考虑 numbers[mid] 和 numbers[right] 比较,可得下面三种情况:

    • numbers[mid] 小于 numbers[right] : right = mid
    • numbers[mid] 大于 numbers[right] : left = mid + 1
    • numbers[mid] 等于 numbers[right] : 因为 numbers[mid] 和 numbers[right] 之间可能并不有序,只能得出最小数一定不在最右位置,所以--right

    代码

    时间复杂度:O(logn)
    空间复杂度:O(1)

    class Solution {
    public:
        int minArray(vector<int>& numbers) {
            int left = 0, right = numbers.size() - 1;        
            while (left < right) {
                int mid = left + (right - left) / 2;
                if (numbers[mid] < numbers[right]) {
                    right = mid;
                } else if (numbers[mid] > numbers[right]) {
                    left = mid + 1;
                } else {
                    --right;
                }
            }
            return numbers[left];
        }
    };
    
  • 相关阅读:
    python中将集合进行切割的方法
    Tomcat单库多实例部署
    (二) storm的基本使用
    工作流表结构解析
    [TCO 2012 Round 3A Level3] CowsMooing (数论,中国剩余定理,同余方程)
    [HEOI 2013 day2] 钙铁锌硒维生素 (线性代数,二分图匹配)
    [HEOI 2013 day2] SAO (树形动态规划)
    关于react和Angular的思考
    angular 中的modal
    从易到难,写一个JavaScript加载器之一
  • 原文地址:https://www.cnblogs.com/galaxy-hao/p/12323975.html
Copyright © 2011-2022 走看看