《剑指offer》旋转数组中的最小数字
本题来自《剑指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》旋转数组中的最小数字的更多相关文章
- 剑指 Offer 03. 数组中重复的数字
剑指 Offer 03. 数组中重复的数字 找出数组中重复的数字. 在一个长度为 n 的数组 nums 里的所有数字都在 0-n-1 的范围内.数组中某些数字是重复的,但不知道有几个数字重复了,也不知 ...
- 菜鸟刷题路:剑指 Offer 03. 数组中重复的数字
剑指 Offer 03. 数组中重复的数字 哈希表/set class Solution { public int findRepeatNumber(int[] nums) { HashSet< ...
- 5.1 剑指 Offer 03. 数组中重复的数字
类型题:剑指 Offer 03. 数组中重复的数字 找出数组中重复的数字.在一个长度为 n 的数组 nums 里的所有数字都在 0-n-1 的范围内.数组中某些数字是重复的,但不知道有几个数字重复了, ...
- Go语言实现:【剑指offer】数组中重复的数字
该题目来源于牛客网<剑指offer>专题. 在一个长度为n的数组里的所有数字都在0到n-1的范围内.数组中某些数字是重复的,但不知道有几个数字是重复的.也不知道每个数字重复几次.请找出数组 ...
- (python)剑指Offer:数组中重复的数字
问题描述 在长度为n的数组中,所有的元素都是0到n-1的范围内. 数组中的某些数字是重复的,但不知道有几个重复的数字,也不知道重复了几次,请找出任意重复的数字. 例如,输入长度为7的数组{2,3,1, ...
- [剑指offer]3.数组中的重复数字
3.数组中的重复数字 题目 找出数组中重复的数字. 在一个长度为 n 的数组 nums 里的所有数字都在 0-n-1 的范围内.数组中某些数字是重复的,但不知道有几个数字重复了,也不知道每个数字重复了 ...
- 【剑指Offer】数组中重复的数字 解题报告(Python & C++)
作者: 负雪明烛 id: fuxuemingzhu 个人博客:http://fuxuemingzhu.cn/ 目录 题目描述 解题方法 Set 快慢指针 日期 题目地址:https://leetcod ...
- 剑指Offer 50. 数组中重复的数字 (数组)
题目描述 在一个长度为n的数组里的所有数字都在0到n-1的范围内. 数组中某些数字是重复的,但不知道有几个数字是重复的.也不知道每个数字重复几次.请找出数组中任意一个重复的数字. 例如,如果输入长度为 ...
- 剑指offer:数组中重复的数字
题目描述: 在一个长度为n的数组里的所有数字都在0到n-1的范围内. 数组中某些数字是重复的,但不知道有几个数字是重复的.也不知道每个数字重复几次.请找出数组中任意一个重复的数字. 例如,如果输入长度 ...
- [剑指Offer] 50.数组中重复的数字
题目描述 在一个长度为n的数组里的所有数字都在0到n-1的范围内. 数组中某些数字是重复的,但不知道有几个数字是重复的.也不知道每个数字重复几次.请找出数组中任意一个重复的数字. 例如,如果输入长度为 ...
随机推荐
- python 07
1.文件操作: f=open(...) 是由操作系统打开文件,那么如果我们没有为open指定编码,那么打开文件的默认编码很明显是操作系统说了算了 操作系统会用自己的默认编码去打开文件,在windows ...
- Aizu 2170 Marked Ancestor
题意:出一颗树,有两种操作:1. mark u 标记结点u2.query u 询问离u最近的且被标记的祖先结点是哪个让你输出所有询问的和. 思路:数据量太小,直接暴力dfs就可以了 #incl ...
- java jvm和android DVM区别
本文转自:http://blog.csdn.net/yujun411522/article/details/45932247 1.Android dvm的进程和Linux的进程, 应用程序的进 ...
- 2018牛客暑期ACM多校训练营第二场(有坑未填)
第二场终于等来学弟 开始(被队友带飞)的开心(被虐)多校之旅 A run A题是一个递推(dp?)+前缀和 因为看数据量比较大 就直接上前缀和了 一个比较简单的递推 没有太多难点 签到题 需要注意 ...
- python中的正则表达式--re模块
参考博客:https://www.cnblogs.com/tina-python/p/5508402.html 这里说一下python的re模块即正则表达式模块,先列出其中涉及到的各种字符和模式等: ...
- Linux登录失败:Too many logins for 'username'.
cat /etc/security/limits.conf #<domain> <type> <item> <value> * - maxlogins ...
- c++学习day2
1. 输入输出 1)控制符 2) 如果要读取所有输入的字符,包括空格和回车,可以有如下两种方法,其中EOF在windows里默认是 ctrl+Z 注:输入字符时,scanf不会跳过空格,输入其他类型数 ...
- 创建一个yum源,rpm安装二进制包
作者:邓聪聪 安装mariadb vi /etc/yum.repos.d/mariadb.repo [mariadb]name=mariadbbaseurl=http://mirrors.neusof ...
- python3+selenium入门16-窗口截图
有时候需要把一些浏览器当前窗口截图下来,比如操作抱错的时候.WebDriver类下.get_screenshot_as_file()方法可窗口截图,需要传入一个截图文件名的路径.window要用\\当 ...
- 题解-BOI 2004 Sequence
Problem bzoj & Luogu 题目大意: 给定序列\(\{a_i\}\),求一个严格递增序列\(\{b_i\}\),使得\(\sum \bigl |a_i-b_i\bigr|\)最 ...