java数据结构与算法刷题-----LeetCode376. 摆动序列

news2025/1/11 10:04:32
java数据结构与算法刷题目录(剑指Offer、LeetCode、ACM)-----主目录-----持续更新(进不去说明我没写完):https://blog.csdn.net/grd_java/article/details/123063846

文章目录

    • 1. 贪心
    • 2. 动态规划
    • 3. 优化版动态规划

在这里插入图片描述

1. 贪心

解题思路:时间复杂度O( n n n),空间复杂度O( 1 1 1)
  1. 将序列想象成一座山峰,这个山峰由石块构成,我们爬这作山峰,可以前进,所以对我们来说,只有前面的石块和后面的石块。
  2. 对我们来说,我们的路线就会有上坡(前面的石块高度(海拔)>后面的石块),下坡(前面的石块海拔<后面的石块),平坡(每块石头的海拔都一样)
  3. 题目就是让我们统计,只保留上下坡的情况下,上坡和下坡依次摆动,共摆动了几次

题目特殊要求:

  1. 只要有一块石头,就可以算一个坡。例如整座山就一块石头[1],那么就算一个摆动。因为爬上这块石头,算上坡,爬下这块石头,算下坡
  2. 有2块石头,但是构成的是平坡,那么只算一个摆动。例如[1,1],两个石头大小一样。爬上第一块石头算上坡,从第一块到第二块石头,只是直走,没有爬坡。然后从第二块石头下去,算下坡。依然是一个上坡,一个下坡,构成一个摆动。
  3. 有两块石头,但是构成的不是平坡,那么算2个摆动。例如[1,2]. 题目要求的。爬上第一块石块算一个摆动,爬下最后一块石头也算一个摆动。
  4. 上坡+平坡+上坡,或者下坡+平坡+下坡,属于没换坡,例如[1,2,2,3],只有两个摆动。爬上1算一个摆动,然后1-2是上坡,2-2是平坡,2-3还是上坡,没有坡度变化,都不算坡。最后爬下最后一块石头3,算一个坡。最终共两个坡。
代码

在这里插入图片描述

class Solution {
    public int wiggleMaxLength(int[] nums) {
        int n = nums.length;
        if (n < 2) return n;//元素小于2个,就只有n个摆动
        int prevdiff = nums[1] - nums[0];//记录当前坡上的差值,是上坡还是下坡,这里不记录平坡
        int ret = prevdiff != 0 ? 2 : 1;//如果是平坡,就只有1个摆动,如果是上坡或者下坡就有2个摆动(题目规定)
        for (int i = 2; i < n; i++) {//之后还有坡,就继续统计
            int diff = nums[i] - nums[i - 1];//看看当前是什么坡,上坡,还是下坡,还是平坡
            //如果是平坡,下次变坡后,必须通过prevdiff来获取,平坡之前是什么坡
            //如果变坡后,和平坡之前不同,才能称为换坡了,否则还是没有换坡,也就不是摆动
            if ((diff > 0 && prevdiff <= 0) || (diff < 0 && prevdiff >= 0)) {//如果换坡了(上坡变下坡,或者下坡变上坡)
                ret++;//换坡了,摆动+1
                prevdiff = diff;//换坡后,prevdiff变成这个坡的坡度(上坡还是下坡)。以方便寻找下一个不同的坡度
            }
        }
        return ret;//将换坡次数返回
    }
}

2. 动态规划

解题思路:时间复杂度O( n n n),空间复杂度O( n n n)

用动态规划来实现上面贪心的思想

动态规划5步曲
  1. DP数组及下标含义
  1. 我们要求出的是到当前石头为止,坡度上下变化的次数。显然dp数组中存储的是坡度摆动次数。要求出谁的?显然是求出,从起始石头到当前石头的。那么下标就是代表当前是哪块石头。但是如何知道前一块是上坡还是下坡呢?我们可以创建两个数组,一个是当前石头上坡时候用,一个下坡时候用。很显然,需要一个下标,两个数组。
  1. 递推公式:dp[]和down[]数组分别用于上坡和下坡
  1. 只要有石头,那么摆动就有1,具体请参考上面的"法一:贪心"的解析。所以up[0] = down[0] = 1.
  2. 从第二块石块开始后面的每一块石头,记为第i块石头。
  3. 我们分别获取up[i-1],down[i-1],它们保存到前一块石头的路径摆动次数。则他俩里面大的那个,一定是上块石头的坡度。因为上一块石头,如果是上坡就放入up,下坡就放入down
  4. 如果第i块石块,是上坡,我们获取Math.max(up[i-1],down[i-1]),如果结果是down[i-1]更大,说明上一块是下坡,而第i块是上坡。则摆动+1.
  5. 依次类推,得到公式:
  1. i是上坡,up[i] = Math.max(up[i-1],down[i-1]+1). down[i] = down[i-1]. 其中down[i] = down[i-1]是因为i石块不是下坡,用不着down,但是down[i]下次会用到,所以将[i-1]的值给[i]

Math.max(up,down)获取的是i-1的坡度,down[i-1]+1,是如果i-1是下坡,那么到i位置的路径摆动次数是i-1的摆动次数+1。因为如果i-1是下坡,而现在i是上坡,down[i-1]就是到前一块路径的摆动次数,i是上坡,要增加一次摆动。等价于如下代码:

int num = Math.max(up[i-1],down[i-1]);
boolean flag = true;//上一个石头默认是上坡
if(num  == down[i-1]) flag = false;//上一个石头是下坡
if(flag) up[i] = up[i-1];//上一个也是上坡,那么摆动不变
else up[i] = down[i-1]+1;//上一个是下坡,那么摆动+1
  1. i是下坡,up[i] = up[i-1]; down[i] = Math.max(up[i-1]+1),down[i-1]
  2. i是平坡,up[i] = up[i-1];down[i] = down[i-1];
  1. dp数组初始化

在这里插入图片描述

  1. 数组遍历顺序:一维数组,无需考虑
代码

在这里插入图片描述

class Solution {
    public int wiggleMaxLength(int[] nums) {
        //和贪心一样
        int n = nums.length;
        if (n < 2) return n;
        //dp数组表示,当前石块i,如果是上坡的,就放入up中,如果是下坡的,就放入down中,然后计算到这块石头为止的摆动
        //up表示上坡,down表示下坡。dp数组的下标表示,第几块石头
        //例如up[1],就是第一块石头如果是上坡,到它为止的路径,它的最大摆动是多少
        //down[3]就是第3块石头,如果是下坡,到它为止的路径,最大摆动是多少
        /**则dp数组的值保存的就是 到第i块石块,摆动的数量 */
        int[] up = new int[n];
        int[] down = new int[n];
        up[0] = down[0] = 1;//有石头就有一个摆动,无论上坡,还是下坡,都算一个摆动。
        for (int i = 1; i < n; i++) {//从第一块石头开始规划坡度
            if (nums[i] > nums[i - 1]) {//如果当前石块是上坡
                //当前石块应该放入上坡UP中,获取前一块石头的信息,前一块如果也是上坡,则摆动数量不变,前一块是下坡,那么摆动+1. 
                //对于我们来说,我们不知道前一块是上坡还是下坡,但是上一块石头的路径摆动,一定放入了dp数组
                //而且一定是最大的,所以up[i-1]和down[i-1],就是获取前一块石头的最大摆动
                up[i] = Math.max(up[i - 1], down[i - 1] + 1);//必须两个一起获取,因为不知道前一块是上坡还是下坡
                //当前石块i是上坡,无法放入下坡,所以对于down来说,只能抛弃这块石头,那么摆动不变
                down[i] = down[i - 1];//让其继承前一块石头的坡度
            } else if (nums[i] < nums[i - 1]) {//如果当前石块是下坡
                up[i] = up[i - 1];//和上坡没有关系,抛弃这块石头
                //获取前一块石头的信息,如果是上坡,那么摆动+1,如果是下坡,摆动不变。
                down[i] = Math.max(up[i - 1] + 1, down[i - 1]);
            } else {//如果是平坡
                up[i] = up[i - 1];//摆动不变
                down[i] = down[i - 1];//摆动不变
            }
        }
        //到最后一块石头石,我们也不知道最后一块是上坡还是下坡。只知道到它为止的摆动,放入了up[n-1]和down[n-1]
        //而且肯定是大的那个
        return Math.max(up[n - 1], down[n - 1]);
    }
}

3. 优化版动态规划

解题思路:时间复杂度O( n n n),空间复杂度O( 1 1 1)

将法二的dp数组优化掉,换成了两个变量
因为法二中,虽然是dp数组,但是我们每次只使用前一个值罢了

代码

在这里插入图片描述

  1. 基于法二,单纯将数组换成变量
class Solution {
    public int wiggleMaxLength(int[] nums) {
        int n = nums.length;
        if (n < 2) return n;
        
        int up = 1, down = 1;//将dp数组,换成两个变量
        for (int i = 1; i < n; i++) {
            if (nums[i] > nums[i - 1]) {//如果是上坡
                up = Math.max(up, down + 1);//看前一个是不是上坡,不是就摆动+1
            } else if (nums[i] < nums[i - 1]) {//如果是下坡
                down = Math.max(up + 1, down);//上一个是上坡,就摆动+1
            }
        }
        return Math.max(up, down);
    }
}

  1. 我们发现,上坡的话,up的值不变,下坡的话,down的值不变.而本次修改up的值后,下次一定是修改down的值。因为我们要找的就是上坡和下坡交替出现,所以根本没必要每次用Math.max()方法
class Solution {
    public int wiggleMaxLength(int[] nums) {
        int n = nums.length;
        if (n < 2) {
            return n;
        }
        int up = 1, down = 1;//将dp数组,换成两个变量
        for (int i = 1; i < n; i++) {
            if (nums[i] > nums[i - 1]) {//如果是上坡
                up = down + 1;//看前一个是不是上坡,不是就摆动+1
            } else if (nums[i] < nums[i - 1]) {
                down = up + 1;
            }
        }
        return Math.max(up, down);
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1522810.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【强化学习笔记一】初识强化学习(定义、应用、分类、性能指标、小车上山案例及代码)

文章目录 第1章 初识强化学习1.1 强化学习及其关键元素1.2 强化学习的应用1.3 强化学习的分类1.3.1 按任务分类1.3.2 按算法分类 1.4 强化学习算法的性能指标1.5 案例&#xff1a;基于Gym库的智能体/环境接口1.5.1 安装Gym库1.5.2 使用Gym库1.5.3 小车上山1.5.3.1 有限动作空间…

软考80-上午题-【面向对象技术3-设计模式】-结构型设计模式03

一、外观模式 1-1、意图 为子系统中的一组接口提供一个一致的界面。 Facade 模式定义了一个高层接口&#xff0c;这个接口使得这一子系统更加容易使用。 1-2、结构 Facade 知道哪些子系统类负责处理请求&#xff1a;将客户的请求代理给适当的子系统对象。Subsvstem classes …

Mock.js了解(Mock就是模拟一个后端,Postman模拟前端)

JSON5 Node.js Vue CLI与Mock.js Jquery与Mock.js Mock与分页

Linux - 线程互斥和互斥锁

文章目录 前言一、为什么要线程互斥原子性 二、互斥锁互斥锁的创建与销毁互斥锁进行互斥 前言 前几节课&#xff0c;我们学习了多线程的基础概念&#xff0c;这节课&#xff0c;我们来对线程互斥和互斥锁的内容进行学习。 一、为什么要线程互斥 首先我们要明白&#xff0c;对…

openGauss学习笔记-244 openGauss性能调优-SQL调优-典型SQL调优点-统计信息调优

文章目录 openGauss学习笔记-244 openGauss性能调优-SQL调优-典型SQL调优点-统计信息调优244.1 统计信息调优244.1.1 统计信息调优介绍244.1.2 实例分析&#xff1a;未收集统计信息导致查询性能差 openGauss学习笔记-244 openGauss性能调优-SQL调优-典型SQL调优点-统计信息调优…

JVM学习-底层字节码的执行过程

目录 1.一个简单的程序分析 2. a&#xff0c;a&#xff0c;a--在JVM中的执行过程 3. 一个好玩的xx 4.方法调用的字节码分析、多态的实现、对象头 5. try-catch-finally的字节码分析 5.1 try-catch 5.2 try-catch-finally 5.3特殊情况 5.3.1 try和finally块中都出现了re…

第18节 动态规划一讲

1假设有排成一行的N个位置记为1~N&#xff0c;N一定大于或等于2 开始时机器人在其中的M位置上(M一定是1~N中的一个) 如果机器人来到1位置&#xff0c;那么下一步只能往右来到2位置&#xff1b; 如果机器人来到N位置&#xff0c;那么下一步只能往左来到N-1位置&#xff1b; 如果…

GiT: Towards Generalist Vision Transformer through Universal Language Interface

GiT: Towards Generalist Vision Transformer through Universal Language Interface 相关链接&#xff1a;arxiv github 关键字&#xff1a;Generalist Vision Transformer (GiT)、Universal Language Interface、Multi-task Learning、Zero-shot Transfer、Transformer 摘要 …

BigDecimal保留两位小数失败问题

文章目录 背景问题解决如何测试代码 背景 测试时发现在线swagger测试会自动处理BigDecimal小数点后面的数字&#xff0c;就是有零的会都给你去掉&#xff0c;比如9.000与9.500到最后都会被swagger处理成9跟9.5。使用postman测是最准的&#xff0c;测出来的就是9.000跟9.500。 …

Rocky Linux 基本工具的安装

1.系统安装后先查看ip地址 ip addr 2.安装net工具 &#xff1a;ifconfig yum install net-tools 3.安装gcc &#xff1b;选择都选 y yum install gcc yum install gcc-c 4.安装tcl yum install -y tcl 5.安装lsof &#xff08;端口查看工具&#xff09; yum install l…

MySQL实现事务隔离的秘诀之锁

在MySQL中&#xff0c;有多种锁类型&#xff0c;我们先了解三种概念的锁&#xff0c;以便对接下来的内容有更好理解。 表级锁&#xff08;Table Lock&#xff09;&#xff1a;对整个表加锁&#xff0c;其他事务无法修改或读取该表的数据&#xff0c;但可以对其他表进行操作。页…

SLAM 算法综述

LiDAR SLAM 其主要思想是通过两个算法&#xff1a;一个高频激光里程计进行低精度的运动估计&#xff0c;即使用激光雷达做里程计计算两次扫描之间的位姿变换&#xff1b;另一个是执行低频但是高精度的建图与校正里程计&#xff0c;利用多次扫描的结果构建地图&#xff0c;细化位…

切面条-蓝桥杯?-Lua 中文代码解题第1题

切面条-蓝桥杯&#xff1f;-Lua 中文代码解题第1题 一根高筋拉面&#xff0c;中间切一刀&#xff0c;可以得到2根面条。 如果先对折1次&#xff0c;中间切一刀&#xff0c;可以得到3根面条。 如果连续对折2次&#xff0c;中间切一刀&#xff0c;可以得到5根面条。 那么&#xf…

【QT入门】VS2019+QT的开发环境配置

声明&#xff1a;该专栏为本人学习Qt知识点时候的笔记汇总&#xff0c;希望能给初学的朋友们一点帮助(加油&#xff01;) 往期回顾&#xff1a; 【QT入门】什么是qt&#xff0c;发展历史&#xff0c;特征&#xff0c;应用&#xff0c;QtCreator-CSDN博客【QT入门】Windows平台下…

Java设计模式 | 设计模式概述和分类

独孤求败五重境界 利剑&#xff08;“凌厉刚猛&#xff0c;无坚不摧&#xff0c;弱冠前以之与河朔群雄争锋。”&#xff09;软剑&#xff08;“紫薇软剑&#xff0c;三十岁前所用&#xff0c;误伤义士不祥&#xff0c;乃弃之深谷。”&#xff09;重剑&#xff08;“重剑无锋&a…

React 实现下拉刷新效果

简介 本文基于react实现下拉刷新效果&#xff0c;在下拉的时候会进入loading状态。 实现效果 效果如上图所示&#xff0c;在下拉到底部时候&#xff0c;会出现loading条&#xff0c;在处理完成后loading条消失。 具体代码 布局 & 逻辑 import {useRef, useState} from …

【IC设计】Verilog线性序列机点灯案例(二)(小梅哥课程)

文章目录 该系列目录&#xff1a;设计目标设计思路RTL 及 Testbench仿真结果存在的问题&#xff1f;改善后的代码RTL代码testbench代码 仿真结果 案例和代码来自小梅哥课程&#xff0c;本人仅对知识点做做笔记&#xff0c;如有学习需要请支持官方正版。 该系列目录&#xff1a;…

javaEE——线程的等待和结束

文章目录 Thread 类及常见方法启动一个线程中断一个线程变量型中断调用 interrupt() 方法来通知观察标志位是否被清除 等待一个线程获取当前线程引用休眠当前线程 线程的状态观察线程的所有状态观察 1: 关注 NEW 、 RUNNABLE 、 TERMINATED 状态的切换 多线程带来的风险为什么会…

Ubuntu 14.04:安装 PaddleOCR 2.3

目录 一、说明 1.1 如何选择版本 1.2 查看 github 中的 PaddleOCR 版本 二、安装 2.1 安装前环境准备 2.2 下载包 2.3 解压 2.4 安装依赖库 异常处理&#xff1a;Read timed out. 2.5 下载推理模型&#xff1a;inference 2.5.1 模型存放位置 2.5.2 模型下载链接 2.5.…

【5G NB-IoT NTN】3GPP R17 NB-IoT NTN介绍

博主未授权任何人或组织机构转载博主任何原创文章&#xff0c;感谢各位对原创的支持&#xff01; 博主链接 本人就职于国际知名终端厂商&#xff0c;负责modem芯片研发。 在5G早期负责终端数据业务层、核心网相关的开发工作&#xff0c;目前牵头6G算力网络技术标准研究。 博客…