题解 | #寻找峰值#

寻找峰值

http://www.nowcoder.com/practice/1af528f68adc4c20bf5d1456eddb080a

题目描述

山峰元素是指其值大于或等于左右相邻值的元素。给定一个输入数组nums,任意两个相邻元素值不相等,数组可能包含多个山峰。找到索引最大的那个山峰元素并返回其索引。

假设 nums[-1] = nums[n] = -∞。

大体思路

题目中的山峰元素定义是要大于等于其紧邻的左右两边的元素就行了,紧接着又说传入的数组中相邻的两个元素值是不相等,所以很自然的就会想到遍历的方式。可以从左往右遍历,也可以从右往左遍历。

因为题目所要求的是索引最大的山峰元素的峰值的,所以从左往右的遍历方式中,我们不能找到一个峰值就停止,要一直往右找完所有山峰元素;但是在从右往左的遍历方式中,我们只要找到第一个山峰元素,就可以停止遍历了,因为第一个找到的山峰元素,就是索引值最大的山峰元素,再往左边找,那索引值只能越来越小。

方法一:从左往右遍历

解题思路

为了防止数组下标越界的问题,就只看数组中第 2 个数字到倒数第 2 个数字这个范围就好了,在遍历的过程要和其左右两边的元素都做比较。先让山峰元素初始化为第 1 个数字,最后再来单独看看最后一个数字是不是山峰元素。

整个过程就很简单很暴力,具体的相关流程见下图示和代码示例。
图片说明

代码示例

import java.util.*;


public class Solution {
    /**
     * 寻找最后的山峰 
     */
    public int solve (int[] nums) {
        // 判断特殊情况
        if (nums == null || nums.length == 0) {
            return -1;
        }

        // 只有一个数字,那他自己就是峰值
        if (nums.length == 1) {
            return 0;
        }

        // 最大的索引
        int maxIndex = 0;

        // 从第2个数字开始往后遍历到倒数第二个数字
        for (int i = 1; i < nums.length - 1; i++) {
            // 只要比前一个数字、后一个数字都大, 那就是峰值
            if (nums[i] > nums[i - 1] && nums[i] > nums[i + 1]) {
                maxIndex = i;
            }
        }

        // 判断最后一个索引是不是最大峰值
        if (nums[nums.length - 1] > nums[nums.length - 2]) {
            maxIndex = nums.length - 1;
        }

        return maxIndex;
    }
}

复杂度分析

时间复杂度:O(N)

整个过程就是从第 2 个数字到倒数第 2 个数字遍历了一遍,只剩 2 个数字在循环里没有遍历到,但但对于 N 这个的变量来说,2 个数字这样的常量所带来的性能消耗可以忽略不计了。所以,总的时间复杂度即为 O(N)。

空间复杂度:O(1)

整个代码中没有出现子再开辟新空间的地方,只有一个用来记录山峰元素当中的最大索引值的整型变量。所以,空间复杂度为 O(1)。

方法二:从右到左遍历

解题思路

在从右往左的遍历过程中,为了防止下标越界,所以遍历顺序及遍历范围就是从右往左遍历最后一个数字到第 2 个数字。在遍历的过程中,只要看是否比他前面的那个数字大就好了,找到了就更新山峰元素的最大索引值,然后退出循环。

因为最后一个数字的右边没有数字,如果在这个过程中找到了一个比他前面那个数字大的元素,那他必定是山峰元素,因为他也比他右边的那个数字大,如果他没有比他右边的那个数字大,那他根本就来不到这一步,因为他右边的那个数字也是一个山峰元素,且索引值要比他的还要大。

在遍历之前,先对山峰元素的最大索引值初始化为 0 ,代表第 1 个数字就是山峰元素,如果在从右到左遍历的过程中,没有找到山峰元素,那么说明第 1 个数字是山峰元素,且是索引值最大的山峰元素。

简单的说,题目要求的是索引最大的山峰元素,所以,可以直接从后往前遍历,只要找到第一个山峰元素,他的索引值就是我们要找的答案。

整个过程可见下图及代码示例:
图片说明

代码示例

import java.util.*;


public class Solution {
    /**
     * 寻找最后的山峰 
     */
    public int solve (int[] nums) {
        // 判断特殊情况
        if (nums == null || nums.length == 0) {
            return -1;
        }

        // 数组只有1个数字,那他自己就是峰值
        if (nums.length == 1) {
            return 0;
        }

        // 最大的索引
        int maxIndex = 0;

        // 从后往前遍历
        for (int i = nums.length - 1; i > 0; i--) {
            // 只要比前面的数大,就是峰值
            if (nums[i] > nums[i - 1]) {
                maxIndex = i;
                break;
            }
        }

        return maxIndex;
    }
}

复杂度分析

时间复杂度:O(N)

仍然是一个遍历的过程,本质上来说和方法一并没有本质的区别,只是遍历的顺序不一样,所以时间复杂度仍然是 O(N) 。

空间复杂度:O(1)

仍然没有开辟新的空间,仍然是只开辟了一个记录山峰元素当中的最大索引值的整型变量。所以,空间复杂度仍然是 O(1)。

全部评论

相关推荐

上周组里招人,我面了六个候选人,回来跟同事吃饭的时候聊起一个让我挺感慨的现象。前三个候选人,算法题写得都不错。第一道二分查找,五分钟之内给出解法,边界条件也处理得干净。第二道动态规划,状态转移方程写对了,空间复杂度也优化了一版。我翻他们的简历,力扣刷题量都在300以上。后三个呢,就有点参差不齐了。有的边界条件没处理好,有的直接说这道题没刷过能不能换个思路讲讲。其中有一个女生,我印象特别深——她拿到题之后没有马上写,而是先问我:“面试官,我能先跟你确认一下我对题目的理解吗?”然后她把自己的思路讲了一遍,虽然最后代码写得不是最优解,但整个沟通过程非常顺畅。这个女生的代码不是最优的,但当我问她“如果这里是线上环境,你会怎么设计’的时候,她给我讲了一套完整的方案——异常怎么处理、日志怎么打、怎么平滑发布。她对这是之前在实习的时候踩过的坑。”我在想LeetCode到底在筛选什么?我自己的经历可能有点代表性。我当年校招的时候,也是刷了三百多道题才敢去面试。那时候大家都刷,你不刷就过不了笔试关。后来工作了,前三年基本没再打开过力扣。真正干活的时候,没人让你写反转链表,也没人让你手撕红黑树。更多的是:这个接口为什么慢了、那个服务为什么OOM了、线上数据对不上了得排查一下。所以后来我当面试官,慢慢调整了自己的评判标准。算法题我还会出,但目的变了。我出算法题,不是想看你能不能背出最优解。而是想看你拿到一个陌生问题的时候,是怎么思考的。你会先理清题意吗?你会主动问边界条件吗?你想不出来的时候会怎么办?你写出来的代码,变量命名乱不乱、结构清不清楚?这些才是工作中真正用得到的能力。LeetCode是一个工具,不是目的。它帮你熟悉数据结构和常见算法思路,这没问题。但如果你刷了三百道题,却说不清楚自己的项目解决了什么问题、遇到了什么困难、你是怎么解决的,那这三百道题可能真的白刷了。所以还要不要刷LeetCode?要刷,但别只刷题。刷题的时候,多问自己几个为什么:为什么用这个数据结构?为什么这个解法比那个好?如果换个条件,解法还成立吗?把刷题当成锻炼思维的方式,而不是背答案的任务。毕竟面试官想看到的,从来不是一台背题机器,而是一个能解决问题的人。
牛客51274894...:意思是光刷力扣还不够卷
AI时代还有必要刷lee...
点赞 评论 收藏
分享
评论
点赞
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务