力扣852(java&python)-山脉数组的峰顶索引(中等)
题目:
符合下列属性的数组 arr 称为 山脉数组 :
- arr.length >= 3
- 存在 i(0 < i < arr.length - 1)使得:
- arr[0] < arr[1] < ... arr[i-1] < arr[i]
- arr[i] > arr[i+1] > ... > arr[arr.length - 1]
- 给你由整数组成的山脉数组 arr ,返回任何满足 arr[0] < arr[1] < ... arr[i - 1] < arr[i] > arr[i + 1] > ... > arr[arr.length - 1] 的下标 i 。
示例 1:
输入:arr = [0,1,0]
输出:1
示例 2:
输入:arr = [0,2,1,0]
输出:1
示例 3:
输入:arr = [0,10,5,2]
输出:1
示例 4:
输入:arr = [3,4,5,1]
输出:2
示例 5:
输入:arr = [24,69,100,99,79,78,67,36,26,19]
输出:2
提示:
- 3 <= arr.length <= 104
- 0 <= arr[i] <= 106
- 题目数据保证 arr 是一个山脉数组
进阶:很容易想到时间复杂度 O(n) 的解决方案,你可以设计一个 O(log(n)) 的解决方案吗?
来源:力扣(LeetCode)
链接:https://leetcode.cn/problems/peak-index-in-a-mountain-array
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。
解题思路:
【二分查找】
其实看到提示中进阶要求设计时间复杂度为O(log(n))的时候就会想到二分查找,加上题目中给定的数组可以看做两个有序的数组,先升序(上山)后降序(下山),目的就是找到升序中最后的那一个数字(山顶)的下标。用二分查找:
- 初始化左右边界:left = 0, right = n - 1,计算出mid
- 循环条件是:left < right,判断的条件就让mid与 mid - 1的数值进行判断
- arr[mid] > = arr[mid - 1]:说明mid还处于上山阶段,此时mid有可能就是山顶,需要往右继续找山顶,搜索区间变为[mid, right],即让left = mid(注意:mid需要向上取整);
- arr[mid] < arr[mid-1]:说明mid已经处于下山阶段,就需要往左寻找山顶,搜索区间变为[left, mid - 1],即让right = mid - 1;
- 循环结束的条件是:left == right,题目中说了 arr 是一个山脉数组,那么最后一定会是山顶,直接返回 left 或者 right 即可。
java代码(left < right):
1 class Solution {
2 public int peakIndexInMountainArray(int[] arr) {
3 int n = arr.length;
4 int left = 0, right = n - 1;
5 while (left < right){
6 int mid = left + (right - left + 1) / 2;
7 if (arr[mid] > arr[mid -1]){
8 left = mid;
9 }else{
10 right = mid - 1;
11 }
12 }
13 return left;
14 }
15 }

python3代码(left <= right):错误代码:
1 class Solution:
2 def peakIndexInMountainArray(self, arr: List[int]) -> int:
3 left, right = 0, len(arr) - 1
4 while left <= right:
5 mid = left + (right - left) // 2
6 if arr[mid] > arr[mid - 1] and arr[mid] > arr[mid + 1]:
7 return mid
8 # arr[mid] > arr[mid - 1]并且arr[mid] <= arr[mid + 1]
9 # mid + 1更大
10 elif arr[mid] > arr[mid - 1]:
11 left = mid + 1
12 else:
13 # arr[mid] > arr[mid + 1]并且arr[mid]<= arr[mid -1]
14 # mid - 1更大
15 right = mid - 1
16 return -1
有一个测试用例过不去:

研究一下它为什么过不去:
[3,9,8,6,4]
①left = 0, right = 4, mid = 2, arr[2] > arr[3]且arr[2] < arr[1]故right = mid -1 = 1;
②left = 0, right = 1,mid = 0, 这时mid -1 = -1已经数组越界了,故为了避免这种判断,arr[mid+1] > arr[mid] 必须写在 arr[mid-1]>arr[mid] 前面判断。跟着正确的,arr[1] > arr[0],left = mid + 1 = 1;
③left = 1,right = 1,mid = 1,arr[0] < arr[1] < arr[2],故直接返回mid = 1。
python3正确代码:
1 class Solution:
2 def peakIndexInMountainArray(self, arr: List[int]) -> int:
3 left, right = 0, len(arr) - 1
4 while left <= right:
5 mid = left + (right - left) // 2
6 if arr[mid] > arr[mid + 1] and arr[mid] > arr[mid - 1]:
7 return mid
8 # arr[mid] > arr[mid + 1]并且arr[mid]<= arr[mid -1]
9 # mid - 1更大
10 elif arr[mid] > arr[mid + 1]:
11 right = mid - 1
12 else:
13 # arr[mid] > arr[mid - 1]并且arr[mid] <= arr[mid + 1]
14 # mid + 1更大
15 left = mid + 1
16 return -1

力扣852(java&python)-山脉数组的峰顶索引(中等)的更多相关文章
- Leetcode之二分法专题-852. 山脉数组的峰顶索引(Peak Index in a Mountain Array)
Leetcode之二分法专题-852. 山脉数组的峰顶索引(Peak Index in a Mountain Array) 我们把符合下列属性的数组 A 称作山脉: A.length >= 3 ...
- 力扣(LeetCode) 852. 山脉数组的峰顶索引
我们把符合下列属性的数组 A 称作山脉: A.length >= 3 存在 0 < i < A.length - 1 使得A[0] < A[1] < ... A[i-1] ...
- LeetCode 852. Peak Index in a Mountain Array (山脉数组的峰顶索引)
题目标签:Binary Search 题目给了我们一组 int array,让我们找到数组的 peak. 利用 binary search, 如果数字比它后面那个数字小,说明还在上坡,缩小范围到右半边 ...
- LeetCode 852. 山脉数组的峰顶索引 (二分)
题目链接:https://leetcode-cn.com/problems/peak-index-in-a-mountain-array/ 我们把符合下列属性的数组 A 称作山脉: A.length ...
- 力扣(LeetCode)寻找数组的中心索引 个人题解
给定一个整数类型的数组 nums,请编写一个能够返回数组“中心索引”的方法. 我们是这样定义数组中心索引的:数组中心索引的左侧所有元素相加的和等于右侧所有元素相加的和. 如果数组不存在中心索引,那么我 ...
- [Swift]LeetCode852. 山脉数组的峰顶索引 | Peak Index in a Mountain Array
Let's call an array A a mountain if the following properties hold: A.length >= 3 There exists som ...
- 力扣 -- 寻找两个有序数组的中位数 Median of Two Sorted Arrays python实现
题目描述: 中文: 给定两个大小为 m 和 n 的有序数组 nums1 和 nums2. 请你找出这两个有序数组的中位数,并且要求算法的时间复杂度为 O(log(m + n)). 你可以假设 nums ...
- 力扣1052. 爱生气的书店老板-C语言实现-中等难度
题目 传送门 文本 今天,书店老板有一家店打算试营业 customers.length 分钟.每分钟都有一些顾客(customers[i])会进入书店,所有这些顾客都会在那一分钟结束后离开. 在某些时 ...
- 力扣(LeetCode)561. 数组拆分 I
给定长度为 2n 的数组, 你的任务是将这些数分成 n 对, 例如 (a1, b1), (a2, b2), ..., (an, bn) ,使得从1 到 n 的 min(ai, bi) 总和最大. 示例 ...
- 力扣Leetcode 面试题51. 数组中的逆序对 - 归并排序
在数组中的两个数字,如果前面一个数字大于后面的数字,则这两个数字组成一个逆序对.输入一个数组,求出这个数组中的逆序对的总数. 示例 1: 输入: [7,5,6,4] 输出: 5 限制: 0 <= ...
随机推荐
- nest.sh 脚本 发布服务
每次发布后端nest 直接执行一个脚本即可 给脚本赋值权限 chomd 777 nest.sh nest.sh 脚本 #!/bin/bash cd /root/gateway-study git pu ...
- DuiLib 一个window的皮肤库 C++ 项目,打包小,比较纯正主流的大制作
https://github.com/duilib/duilib 从 火柴 那个软件 发现的这个库
- 在运行程序是出现sh: 行 1: cls: 未找到命令
在运行程序是出现sh: 行 1: cls: 未找到命令 原因是system("cls");--这是在程序中调用系统命令,但是linux识别不了.功能是清除当前的终端显示数据.找到l ...
- TLSR8258方案开发之BLE协议接口代码解析
一 前言 这里的代码是在原厂基础上修改了不少.虽然代码复杂了不少,但是逻辑也清晰了不少. 二 广播协议 想要熟悉并修改ble的广播协议和内容,请查阅结构体: static const attribu ...
- pip 安装requirements.txt 的问题
用新环境 在进行pip 安装的时候, 如果出现不进行安装 ,但是不报错就是满足条件,这个时候重新起一个shell,然后进行pip的安装.
- javascript之call用法实例
call方法: 调用一个对象的一个方法,以另一个对象替换当前对象. 直接上代码: js例子:在A类中调用B类数据 function ClassA(){ this.name = 'ClassA' ...
- [置顶]
java.io.IOException: No such file or directory解决方案之权限问题
先贴出异常信息: java.io.IOException: No such file or directory at java.io.UnixFileSystem.createFileExclusiv ...
- webserver总结
可设置参数:连接池最大连接数,最大线程数,任务队列最大值,timeslot epoll epoll监听listen_fd(接受新客户端和断开连接), pipefd(将信号输入到管道用epoll统一管理 ...
- 云游长江大桥,3DCAT实时云渲染助力打造沉浸化数字文旅平台
南京长江大桥是中国第一座自主设计建造的双层公路铁路桥,也是世界上最早的双层公路铁路桥之一.它不仅是一座桥梁,更是一座历史文化的见证者和传承者.它见证了中国人民的智慧和奋斗,承载了中国社会的变迁和发展. ...
- 记录--uniapp登录流程详解uni.login
这里给大家分享我在网上总结出来的一些知识,希望对大家有所帮助 uni.login(OBJECT)登录 H5平台登陆注意事项: 微信内嵌浏览器运行H5版时,可通过js sdk实现微信登陆,需要引入一个单 ...