• 【剑指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 动图、动画制作 —— moviepy、matplotlib.animation
    理解 t-SNE (Python)
    理解 t-SNE (Python)
    numpy 辨异(二) —— np.identity()/np.eye()
    numpy 辨异(二) —— np.identity()/np.eye()
    走遍中国 —— 大西南
    走遍中国 —— 大西南
    学习 NLP(一)—— TF-IDF
    我对互联网营销时代的发展认知
    [置顶] MyEclipse下安装插件方法(properties文件编辑器Propedit为例)
  • 原文地址:https://www.cnblogs.com/galaxy-hao/p/12323975.html
Copyright © 2020-2023  润新知