目录
NC107 寻找峰值
描述
给定一个长度为n的数组nums,请你找到峰值并返回其索引。数组可能包含多个峰值,在这种情况下,返回任何一个所在位置即可。
1.峰值元素是指其值严格大于左右相邻值的元素。严格大于即不能有等于
2.假设 nums[-1] = nums[n] = 负无穷
3.对于所有有效的 i 都有 nums[i] != nums[i + 1]
4.你可以使用O(logN)的时间复杂度实现此问题吗?、
示例1
输入:[2,4,1,2,7,8,4]
返回值:1
说明:4和8都是峰值元素,返回4的索引1或者8的索引5都可以
示例2
输入:[1,2,3,1]
返回值:2
说明:3 是峰值元素,返回其索引 2
思路
1---暴力解法,
利用for循环来写,时间复杂度是o(n),nums[i]与nums[i-1]和nums[i+1]比较,大于再输出,但是这样的话会有部分案例答案错误,超时了
所以更加推荐去使用二分查找的方式来写这道题,我们也可以看题目中的你可以使用O(logN)的时间复杂度实现此问题吗,这就在暗示我们使用二分方法来写的
2---二分查找
因为题目将数组边界看成最小值,而我们只需要找到其中一个波峰,因此只要不断地往高处走,一定会有波峰。那我们可以每次找一个标杆元素,将数组分成两个区间,每次就较高的一边走,因此也可以用分治来解决,而标杆元素可以选择区间中点。
具体做法:
- step 1:二分查找首先从数组首尾开始,每次取中间值,直到首尾相遇。
- step 2:如果中间值的元素大于它右边的元素,说明往右是向下,我们不一定会遇到波峰,但是那就往左收缩区间。这是为什么left=mid+1的原因
- step 3:如果中间值大于右边的元素,说明此时往右是向上,向上一定能有波峰,那我们往右收缩区间。
- step 4:最后区间收尾相遇的点一定就是波峰。
int findPeakElement(int* nums, int numsLen ) { // write code here int left = 0; int right = numsLen - 1; while (left < right) { int mid =(right-left)/2+left; if(nums[mid] < nums[mid + 1]) left = mid+1;//这里left如果是mid的话,就会错过峰值了 else right = mid; } return left; }