剑指Offer面试题:6.旋转数组中的最小数字

一 题目:旋转数组中的最小数字

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

  这道题最直观的解法并不难,从头到尾遍历数组一次,我们就能找出最小的元素。这种思路的时间复杂度显然是O(n)。但是这个思路没有利用输入的旋转数组的特性,肯定达不到面试官的要求

  我们注意到旋转之后的数组实际上可以划分为两个排序的子数组,而且前面的子数组的元素都大于或者等于后面子数组的元素。我们还注意到最小的元素刚好是这两个子数组的分界线。在排序的数组中我们可以用二分查找法实现O(logn)的查找

二 代码实现

#include "stdio.h"
#include <iostream>
using namespace std;

#include <assert.h>

// 遍历序列,找到最小值
int NormalFindMinVal(int *pStart, int *pEnd)
{
    assert(pStart != NULL && pEnd != NULL);
    int min = *pStart++;
    while(pEnd - pStart >= 0)
    {
        if (min > *pStart)
        {
            min = *pStart;
        }

        pStart++;
    }
    return min;
}

int SearchMinInRotateaArr_1(int *pStart, int *pEnd)
{
    if (pStart == NULL || pEnd == NULL)
    {
        return -1;
    }
    if (pEnd - pStart == 1)
    {
        return *pEnd;
    }

    int *pMid = NULL;

    pMid = pStart + (pEnd - pStart) / 2;

    if (*pMid >= *pStart && *pMid <= *pEnd)
    {
        return NormalFindMinVal(pStart, pEnd);
    }
    if (*pMid >= *pStart)
    {
        pStart = pMid;
    }
    else if(*pMid <= *pEnd)
    {
        pEnd = pMid;
    }

    return SearchMinInRotateaArr_1(pStart, pEnd);
}

// 找到旋转数组中的最小数字
int SearchMinInRotateaArr(int arr[], int nLen)
{
    assert(arr != NULL && nLen > 0);

    int *pStart,*pEnd;
    pStart = arr;
    pEnd = pStart + nLen -1;

    return SearchMinInRotateaArr_1(pStart, pEnd);
}

int a[] = {3,4,5,1,2};
int b[] = {1,1,1,0,1,1};
int c[] = {1,2,3,4,5};
void main()
{
    int data = SearchMinInRotateaArr(a,sizeof(a)/sizeof(a[0]));
    cout << data << endl;
    data = SearchMinInRotateaArr(b,sizeof(b)/sizeof(b[0]));
    cout << data << endl;
    data = SearchMinInRotateaArr(c,sizeof(c)/sizeof(c[0]));
    cout << data << endl;
    return;
}

原文地址:https://www.cnblogs.com/xiaobingqianrui/p/8831080.html

时间: 2024-08-04 20:00:53

剑指Offer面试题:6.旋转数组中的最小数字的相关文章

【剑指offer】六,旋转数组中的最小数字

题目描述 把一个数组最开始的若干个元素搬到数组的末尾,我们称之为数组的旋转.输入一个递增排序的数组的一个旋转,输出旋转数组的最小元素.例如数组{3,4,5,1,2}为{1,2,3,4,5}的一个旋转,该数组的最小值为1. 分析:数组旋转后会变为一个局部有序的数组,分为两部分,都是递增的.查找最小元素,我们首先想到的时冒泡排序一遍,或者一趟选择排序.或者直接找出a[i]<a[i-1]的点即为要查找的点.也可利用折半查找,判断最小值所在的一侧. 1 import java.util.ArrayLis

剑指Offer 面试题36:数组中的逆序对及其变形(Leetcode 315. Count of Smaller Numbers After Self)题解

剑指Offer 面试题36:数组中的逆序对 题目:在数组中的两个数字,如果前面一个数字大于后面的数字,则这两个数字组成一个逆序对.输入一个数组,求出这个数组中的逆序对的总数. 例如, 在数组{7,5,6,4}中,一共存在5个逆序对,分别是(7,6),(7,5),(7,4),(6,4)和(5,4),输出5. 提交网址: http://www.nowcoder.com/practice/96bd6684e04a44eb80e6a68efc0ec6c5?tpId=13&tqId=11188 或 htt

剑指offer面试题29:数组中出现次数超过一半的数字

题目:数组中有一个数字出现的次数超过数组长度的一般,请找出这个数字,例如输入一个长度为9的数组(1,2,3,2,2,2,5,4,2,).由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2. 个人第一眼想法是通过一个sort函数,再判断中间那数出现次数,只要出现多于n/2,就直接输出. 一般来说,最为直观的算法面试官都不会满意,那么有没有更优的算法呢? 这种算法是受快速排序算法的启发.在随机快速排序算法中,我们现在数组中随机选择一个数字,然后调整数组中数字的顺序,使得比选中的数字小的数字

剑指offer面试题8——旋转数组的最小数字

题目1386:旋转数组的最小数字 时间限制:1 秒 内存限制:32 兆 特殊判题:否 提交:6708 解决:1505 题目描述: 把一个数组最开始的若干个元素搬到数组的末尾,我们称之为数组的旋转.输入一个递增排序的数组的一个旋转,输出旋转数组的最小元素.例如数组{3,4,5,1,2}为{1,2,3,4,5}的一个旋转,该数组的最小值为1. 输入: 输入可能包含多个测试样例,对于每个测试案例, 输入的第一行为一个整数n(1<= n<=1000000):代表旋转数组的元素个数. 输入的第二行包括n

剑指Offer 面试题40:数组中只出现一次的两个数 题解

面试题40:数组中只出现一次的两个数 提交网址:  http://www.nowcoder.com/practice/e02fdb54d7524710a7d664d082bb7811?tpId=13&tqId=11193 或 http://ac.jobdu.com/problem.php?pid=1351 题目:一个整型数组里除了两个数字之外,其他的数字都出现了两次.请写程序找出这两个只出现一次的数字.要求时间复杂度是O(n),空间复杂度是O(1). 牛客网OJ的输入输出接口是:void Fin

剑指offer——面试题40:数组中只出现一次的数字

题目: 一个整型数组里除了两个数字之外,其他的数字都出现了两次.请写程序找出这两个只出现一次的数字. 思路: (1)首先考虑数组中若是只有一个数字是出现一次,其他都是出现两次的情况,在这种情况下,将所有的 数字都位异或运算(^),因为相同的数字异或结果为0,所以最后地到的结果就是那个出现一次的数. (2)下面考虑本题出现有两个出现一次,其他的都是出现一次,这种情况下,可以将所有的异或一遍, 那么最后得到的结果为这两个出现一次的数的异或的结果. 然后找到这个数的从右为0数起的第一个不为0的位,这样

剑指offer——面试题36:数组中的逆序对(归并排序)

题目: 在数组中的两个数字,如果前面一个数字大于后面的数字,则这两个数字组成一个逆序对.输入一个数组,求出这个数组中的逆序对的总数. 思路: 采用归并排序的思想,将数组中的元素分割成子数组,先统计出子数组里的逆序对的个数.同时将这些子数组的数字排成有序的 慢慢往多的合并,在合并的过程中一面统计逆序对的个数,一面合并成一个数组时将里面排好序. 和合并排序类似,时间复杂度为O(nlogn) 下面是详细代码: #include<iostream> #include<vector> usi

剑指offer——面试题29:数组中出现次数超过一半的数

题目描述: 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字. 例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2. 输入: 每个测试案例包括2行: 第一行输入一个整数n(1<=n<=100000),表示数组中元素的个数. 第二行输入n个整数,表示数组中的每个元素,这n个整数的范围是[1,1000000000]. 输出: 对应每个测试案例,输出出现的次数超过数组长度的一半的数,如果没有输出-1. 解法一:基

剑指offer(二十六)之数组中重复的数字

题目描述 在一个长度为n的数组里的所有数字都在0到n-1的范围内. 数组中某些数字是重复的,但不知道有几个数字是重复的.也不知道每个数字重复几次.请找出数组中任意一个重复的数字. 例如,如果输入长度为7的数组{2,3,1,0,2,5,3},那么对应的输出是重复的数字2或者3. 思路一: 用冒泡的思想,当遇到相等的元素的时候,将它放到duplication[0]当中,并将标志tag,改为true <span style="font-family:SimSun;font-size:18px;&