本题来自《剑指offer》 旋转数组中的最小数字

题目:

  把一个数组最开始的若干个元素搬到数组的末尾,我们称之为数组的旋转。 输入一个非减排序的数组的一个旋转,输出旋转数组的最小元素。 例如数组{3,4,5,1,2}为{1,2,3,4,5}的一个旋转,该数组的最小值为1。 NOTE:给出的所有元素都大于0,若数组大小为0,请返回0。

思路:

  从问题本身入手,可以看出,数据局部有序,思考数据中是否重复。属于查找范畴,顺序查找肯定可以,需要O(n)时间,数据有序可以考虑二分查找为log(n)时间内。

  若数据不重复:二分法查找

    可以局部看成两个有序的数组A、B。p1指向首,p2指向尾:

      如果中间值大于p1的值,说明最小值在B中,将p1指向中间值。

      如果中间值小于p2的值,说明最小值A中,将p2指向中间值。

      如果中间值和p1和p2的值相等,则采用下面的方法,顺序查找。

      终止条件是,当p1和p2相邻,最小值便是p2指向的值。返回即可

  若数据重复:顺序查找

    假设第一个为最小的元素,开始遍历,后面凡是小于该值的便保存,最后返回。

C++ Code:

class Solution {
public:
int minNumberInRotateArray(vector<int> rotateArray) {
int p1 = 0;
int p2 = rotateArray.size()-1;
int mid = p1;
while (rotateArray[p1] >= rotateArray[p2]){ //循环条件首值应该大于后值
if (p2 - p1 == 1){ //当首尾相邻时候
mid = p2; //那么最小值是p2的方向
break;
}
mid = (p1 + p2)/2; //中间值为加权平均
if (rotateArray[mid]==rotateArray[p1]&&rotateArray[mid]==rotateArray[p2]){
return minInorder(rotateArray); //如果三值相等,则采用顺序查找法
}
if (rotateArray[mid] >= rotateArray[p1]){ //如果中间值大于首值,说明最小值在后面的数组中
p1 = mid; //手指针后移
}
else if (rotateArray[mid] <= rotateArray[p2]){ //如果中间值小于末值,说明最小值在前面的数组中
p2 = mid; //将末值前移
}
}
return rotateArray[mid];
}
int minInorder(vector<int> rotateArray){ //顺序查找法为O(n)
int result = rotateArray[0]; //假设最小值是首值
for (int i = 1;i<rotateArray.size();i++){
if (result >= rotateArray[i]){ //如果当找到小于该值时候,便保存
result = rotateArray[i];
}
}
return result;
}
};

Python Code:

# -*- coding:utf-8 -*-
class Solution:
def minNumberInRotateArray(self, rotateArray):
# write code here
p1 = 0 #首地址
p2 = len(rotateArray)-1 #尾地址
mid = p1 #起初直接将第一个元素作为中间值,如果第一个值小于最后一个值,便直接返回
while rotateArray[p1] >= rotateArray[p2]: #循环条件是,起始的值大于末值
if p2 - p1 == 1: #终止条件是,当起始和末值相邻近
mid = p2 #将末值赋给中间返回
break
mid = (p1 + p2) // 2 #中间值是首末加权平均,python3中//是整除
if rotateArray[mid] == rotateArray[p1] and rotateArray[mid] == rotateArray[p2]:
return self.minInorder(rotateArray) #当三个值全部相等时候,说明有重复的值,采用顺序查找
if rotateArray[mid] >= rotateArray[p1]: #当中间值大于首值时候,说明最小值在后面的数组中
p1 = mid #便将首地址后移
elif rotateArray[mid] <= rotateArray[p2]: #如果中间值小于末值时候,说明最小值在前面的数组中
p2 = mid #便将末值前移
return rotateArray[mid] #其中间值便是最小的值
def minInorder(self,rotateArray): #顺序查找代码
result = rotateArray[0] #假设第一个值是最小值
if len(rotateArray): #特殊测试,测试是否为空
for i in range(len(rotateArray)-1):
if rotateArray[i] > rotateArray[i+1]: #将小于它的值保存起来
result = rotateArray[i+1]
break
return result

总结:

  从问题本身入手,明确要处理的问题是什么,要求什么样,时间和空间如何转换。以求最优。

《剑指offer》旋转数组中的最小数字的更多相关文章

  1. 剑指 Offer 03. 数组中重复的数字

    剑指 Offer 03. 数组中重复的数字 找出数组中重复的数字. 在一个长度为 n 的数组 nums 里的所有数字都在 0-n-1 的范围内.数组中某些数字是重复的,但不知道有几个数字重复了,也不知 ...

  2. 菜鸟刷题路:剑指 Offer 03. 数组中重复的数字

    剑指 Offer 03. 数组中重复的数字 哈希表/set class Solution { public int findRepeatNumber(int[] nums) { HashSet< ...

  3. 5.1 剑指 Offer 03. 数组中重复的数字

    类型题:剑指 Offer 03. 数组中重复的数字 找出数组中重复的数字.在一个长度为 n 的数组 nums 里的所有数字都在 0-n-1 的范围内.数组中某些数字是重复的,但不知道有几个数字重复了, ...

  4. Go语言实现:【剑指offer】数组中重复的数字

    该题目来源于牛客网<剑指offer>专题. 在一个长度为n的数组里的所有数字都在0到n-1的范围内.数组中某些数字是重复的,但不知道有几个数字是重复的.也不知道每个数字重复几次.请找出数组 ...

  5. (python)剑指Offer:数组中重复的数字

    问题描述 在长度为n的数组中,所有的元素都是0到n-1的范围内. 数组中的某些数字是重复的,但不知道有几个重复的数字,也不知道重复了几次,请找出任意重复的数字. 例如,输入长度为7的数组{2,3,1, ...

  6. [剑指offer]3.数组中的重复数字

    3.数组中的重复数字 题目 找出数组中重复的数字. 在一个长度为 n 的数组 nums 里的所有数字都在 0-n-1 的范围内.数组中某些数字是重复的,但不知道有几个数字重复了,也不知道每个数字重复了 ...

  7. 【剑指Offer】数组中重复的数字 解题报告(Python & C++)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客:http://fuxuemingzhu.cn/ 目录 题目描述 解题方法 Set 快慢指针 日期 题目地址:https://leetcod ...

  8. 剑指Offer 50. 数组中重复的数字 (数组)

    题目描述 在一个长度为n的数组里的所有数字都在0到n-1的范围内. 数组中某些数字是重复的,但不知道有几个数字是重复的.也不知道每个数字重复几次.请找出数组中任意一个重复的数字. 例如,如果输入长度为 ...

  9. 剑指offer:数组中重复的数字

    题目描述: 在一个长度为n的数组里的所有数字都在0到n-1的范围内. 数组中某些数字是重复的,但不知道有几个数字是重复的.也不知道每个数字重复几次.请找出数组中任意一个重复的数字. 例如,如果输入长度 ...

  10. [剑指Offer] 50.数组中重复的数字

    题目描述 在一个长度为n的数组里的所有数字都在0到n-1的范围内. 数组中某些数字是重复的,但不知道有几个数字是重复的.也不知道每个数字重复几次.请找出数组中任意一个重复的数字. 例如,如果输入长度为 ...

随机推荐

  1. 【译】第三篇 SQL Server安全主体和安全对象

    本篇文章是SQL Server安全系列的第三篇,详细内容请参考原文. 一般来说,你通过给主体分配对象的权限来实现SQL Server上的用户与对象的安全.在这一系列,你会学习在SQL Server实例 ...

  2. hibernate多表操作

    一.表之间的关系 1.一对一 2.一对多 3.多对多 二.表之间关系建表原则 1.一对多:在多的一方创建一个外键,指向一的一方的主键 2.多对多:创建一个中间表,中间表至少有两个字段,分别作为外键指向 ...

  3. Sprng4之JDBC--很原始的使用方法

    \[www.dev1234.com]一头扎进Spring4视频教程\一头扎进Spring4源码\[www.java1234.com]<一头扎进Spring4>第九讲 源码\Spring40 ...

  4. 黑马程序员_超全面的JavaWeb视频教程vedio--.ppt,.pptx,.doc,.txt,.docx列表

    \JavaWeb视频教程_day1-资料源码\day01-html目录.txt;\JavaWeb视频教程_day1-资料源码\PPT\HTML.ppt;\JavaWeb视频教程_day1-资料源码\资 ...

  5. JS 如何将 HTML 页面导出为多页 PDF

    参考链接:https://blog.csdn.net/pwc1996/article/details/70141383

  6. 【转载】论文笔记系列-Tree-CNN: A Deep Convolutional Neural Network for Lifelong Learning

    一. 引出主题¶ 深度学习领域一直存在一个比较严重的问题——“灾难性遗忘”,即一旦使用新的数据集去训练已有的模型,该模型将会失去对原数据集识别的能力.为解决这一问题,本文提出了树卷积神经网络,通过先将 ...

  7. launch 文件的写法

    1. launch文件的写法 ❀标签          ☺<node> 启动一个节点          ☺ <param> 设置参数服务器的参数          ☺ < ...

  8. SpringCloud Netflix Ribbon(负载均衡)

    ⒈Ribbon是什么? Spring Cloud Ribbon是基于Netflix Ribbon实现的一套客户端负载均衡工具. Ribbon是Netflix发布的开源项目,主要功能是提供客户端的软件负 ...

  9. Centos7.5 VMtools的安装与卸载

    一.安装 1.自带tools:  选择VMware工具栏 => 虚拟机 => 安装VMtools 2.挂载光驱 3.tar -zxvf VMwareTools-10.3.2-9925305 ...

  10. win7系统查看硬盘序列号步骤

    1.在开始那里输入cmd,打开命令窗口: 2.输入diskpart,按enter键,进入底盘查看选项: 3.输入list disk,按回车键: list disk:查看电脑上有几块硬盘: 输入sele ...