【LeetCode: 152. 乘积最大子数组 | 暴力递归=>记忆化搜索=>动态规划】

news2024/12/29 2:27:34

在这里插入图片描述

🚀 算法题 🚀

🌲 算法刷题专栏 | 面试必备算法 | 面试高频算法 🍀
🌲 越难的东西,越要努力坚持,因为它具有很高的价值,算法就是这样✨
🌲 作者简介:硕风和炜,CSDN-Java领域新星创作者🏆,保研|国家奖学金|高中学习JAVA|大学完善JAVA开发技术栈|面试刷题|面经八股文|经验分享|好用的网站工具分享💎💎💎
🌲 恭喜你发现一枚宝藏博主,赶快收入囊中吧🌻
🌲 人生如棋,我愿为卒,行动虽慢,可谁曾见我后退一步?🎯🎯

🚀 算法题 🚀

在这里插入图片描述

🍔 目录

    • 🚗 知识回顾
    • 🚩 题目链接
    • ⛲ 题目描述
    • 🌟 求解思路&实现代码&运行结果
      • ⚡ 暴力递归
        • 🥦 求解思路
        • 🥦 实现代码
        • 🥦 运行结果
      • ⚡ 记忆化搜索
        • 🥦 求解思路
        • 🥦 实现代码
        • 🥦 运行结果
      • ⚡ 动态规划
        • 🥦 求解思路
        • 🥦 实现代码
          • 实现方式1
          • 实现方式2
        • 🥦 运行结果
    • 💬 共勉

🚗 知识回顾

大家再看这道题目之前,可以先去看一下我之前写过的一篇关于连续子数组算法题的博客,再看这个题目就更容易理解了。
博客的地址放到这里了,可以先去学习一下这到题目。

  • 【LeetCode: 53. 最大子数组和 | 暴力递归=>记忆化搜索=>动态规划 | 分治法 】
  • 【LeetCode: 718. 最长重复子数组 | 暴力递归=>记忆化搜索=>动态规划】

🚩 题目链接

  • 152. 乘积最大子数组

⛲ 题目描述

给你一个整数数组 nums ,请你找出数组中乘积最大的非空连续子数组(该子数组中至少包含一个数字),并返回该子数组所对应的乘积。

测试用例的答案是一个 32-位 整数。

子数组 是数组的连续子序列。

示例 1:

输入: nums = [2,3,-2,4]
输出: 6
解释: 子数组 [2,3] 有最大乘积 6。
示例 2:

输入: nums = [-2,0,-1]
输出: 0
解释: 结果不能为 2, 因为 [-2,-1] 不是子数组。

提示:

1 <= nums.length <= 2 * 104
-10 <= nums[i] <= 10
nums 的任何前缀或后缀的乘积都 保证 是一个 32-位 整数

🌟 求解思路&实现代码&运行结果


⚡ 暴力递归

🥦 求解思路

  1. 子数组类型的题目我们同样可以从以某一位个位置结尾的情况是什么样的角度来进行一个思考。
  2. 这个题目区别于最大子数组和不同的是乘积最大,为什么说这个不同呢?因为当前位置的最大值可以是正数*正数,也可以是负数*负数得来的,那么某一个位置结束的最大值的情况就来源于以下三种情况:1. 前一个位置最大的值*当前位置 2.就是当前位置的值 3. 前一个位置最小的值*当前位置的数。
  3. 上述描述的只是一个过程,接下来我们还需要通过不断的迭代所有位置结尾的位置,得到最大的值。
  4. 下面就是代码的具体实现过程。

🥦 实现代码

class Solution {
    
    public int maxProduct(int[] nums) {
        int n=nums.length;
        int max=Integer.MIN_VALUE;
        for(int i=0;i<n;i++){
            max=Math.max(max,process(i,nums));
        }
        return max;
    }

    public int process(int i,int[] nums){
        if(i<0) return 1;
        return Math.max(process(i-1,nums)*nums[i],Math.max(process1(i-1,nums)*nums[i],nums[i]));
    }

    public int process1(int i,int[] nums){
        if(i<0) return 1;
        return Math.min(process1(i-1,nums)*nums[i],Math.min(process(i-1,nums)*nums[i],nums[i]));
    }
}

🥦 运行结果

时间超限了,不要紧哦,我还有锦囊妙计!

在这里插入图片描述


⚡ 记忆化搜索

🥦 求解思路

  1. 根据我们递归的分析,在递归的过程中会产生重复的子过程,所以我们想到了加一个缓存表,也就是我们的记忆化搜索。

🥦 实现代码

class Solution {
    
    int[] dp1;
    int[] dp2;

    public int maxProduct(int[] nums) {
        int n=nums.length;
        dp1=new int[n];
        dp2=new int[n];
        Arrays.fill(dp1,-1);
        Arrays.fill(dp2,-1);
        int max=Integer.MIN_VALUE;
        for(int i=0;i<n;i++){
            max=Math.max(max,process(i,nums));
        }
        return max;
    }

    public int process(int i,int[] nums){
        if(i<0) return 1;
        if(dp1[i]!=-1) return dp1[i];
        return dp1[i]=Math.max(process(i-1,nums)*nums[i],Math.max(process1(i-1,nums)*nums[i],nums[i]));
    }

    public int process1(int i,int[] nums){
        if(i<0) return 1;
        if(dp2[i]!=-1) return dp2[i];
        return dp2[i]=Math.min(process1(i-1,nums)*nums[i],Math.min(process(i-1,nums)*nums[i],nums[i]));
    }
}

🥦 运行结果

在这里插入图片描述


⚡ 动态规划

🥦 求解思路

  1. 按照我们之前递归和记忆化搜索的思路,通过动态规划实现出来。

🥦 实现代码

注意:只是代码的实现方式不同,核心的原理是一致的。

实现方式1
class Solution {
    
    int[] dp1;
    int[] dp2;
    public int maxProduct(int[] nums) {
        int n=nums.length;
        dp1=new int[n];
        dp2=new int[n];
        dp1[0]=dp2[0]=nums[0];
        int max=nums[0];
        for(int i=1;i<n;i++){
            dp1[i]=Math.max(dp1[i-1]*nums[i],Math.max(dp2[i-1]*nums[i],nums[i]));
            dp2[i]=Math.min(dp2[i-1]*nums[i],Math.min(dp1[i-1]*nums[i],nums[i]));
            max=Math.max(max,dp1[i]);
        }
        return max;
    }
}
实现方式2
class Solution {
    
    int[] dp1;
    int[] dp2;

    public int maxProduct(int[] nums) {
        int n=nums.length;
        dp1=new int[n+1];
        dp2=new int[n+1];
        dp1[n]=dp2[n]=1;
        int max=Integer.MIN_VALUE;
        for(int i=n-1;i>=0;i--){
            dp1[i]=Math.max(dp1[i+1]*nums[i],Math.max(dp2[i+1]*nums[i],nums[i]));
            dp2[i]=Math.min(dp2[i+1]*nums[i],Math.min(dp1[i+1]*nums[i],nums[i]));
            max=Math.max(max,dp1[i]);
        }
        return max;
    }
}

🥦 运行结果

在这里插入图片描述


💬 共勉

最后,我想和大家分享一句一直激励我的座右铭,希望可以与大家共勉!

在这里插入图片描述

在这里插入图片描述

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

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

相关文章

双语|中国和印度仍然主导着美国的国际学者领域

由美国国务院教育和文化事务局支持的国际教育学会期刊《门户开放》&#xff0c;调查了2021-2022赴美的国际学者来源情况&#xff0c;发表文章&#xff0c;“China and India still dominate international scholars field in US”&#xff08;中国和印度仍然主导着美国的国际学…

初中级测试工程师,软件测试面试题总结大全(功能/接口/自动化测试)你要的都有...

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 一般软件测试的面…

安全防御 IPsec VPN

目录 1.什么是数据认证&#xff0c;有什么用&#xff0c;有哪些实现的技术手段&#xff1f; 2.什么是身份认证&#xff0c;有什么用&#xff0c;有哪些实现的技术手段&#xff1f; 3.什么是VPN技术&#xff1f; 4.VPN技术有哪些分类&#xff1f; 5.IPsec技术能够提供哪些安…

linux中静态库与动态库

linux中静态库与动态库 1. 静态库静态库的制作&#xff1a;静态库的使用&#xff1a; 2. 动态库动态库的制作&#xff1a;动态库的使用&#xff1a; linux中静态库与动态库的区别 1. 静态库 静态库的制作&#xff1a; gcc add.c mult.c -c //这样就生成add.o mult.o目标文件 …

【深度学习】基于MindSpore和pytorch的Softmax回归及前馈神经网络

1 实验内容简介 1.1 实验目的 &#xff08;1&#xff09;熟练掌握tensor相关各种操作&#xff1b; &#xff08;2&#xff09;掌握广义线性回归模型&#xff08;logistic模型、sofmax模型&#xff09;、前馈神经网络模型的原理&#xff1b; &#xff08;3&#xff09;熟练掌…

UBUNTU下NFS配置(用于嵌入式开发)

1. NFS简介 NFS&#xff08;Network File System&#xff09;即网络文件系统&#xff0c;是FreeBSD支持的文件系统中的一种&#xff0c;它允许网络中的计算机之间共享资源。在NFS的应用中&#xff0c;本地NFS的客户端应用可以透明地读写位于远端NFS服务器上的文件&#xff0c;就…

低代码(九)低代码平台后设计一:模型驱动

我们先看一下汽车的基本构造&#xff0c;由车身、发动机、方向盘等多个零部件构成&#xff0c;因为它是一个工业产品&#xff0c;有实物存在&#xff0c;摸得着看得见&#xff0c;所以大家很容易理解。日本丰田汽车是如何做到自动化流水线生产的&#xff0c;本质上是把产品xBOM…

BufferedOutputStream,BufferedInputStream是字节流,对象处理流,序列化,输入输出流,转换流

BufferedInputStream字节输入流 意思就是InputStream类及其子类都能以参数的形式放到BufferedInputStream构造器的参数 package com.hspedu.outputstream_;import java.io.*;/*** author 韩顺平* version 1.0* 演示使用BufferedOutputStream 和 BufferedInputStream使用* 使用他…

数据挖掘:心脏病预测(测评指标;EDA)

目录 一、前期准备 二、实战演练 2.1分类指标评价计算示例 2.2数据探索性分析&#xff08;EDA&#xff09; 2.2.1 导入函数工具箱 2.2.2 查看数据信息等相关数据 判断数据缺失和异常 数字特征相互之间的关系可视化 类别特征分析&#xff08;箱图&#xff0c;小提琴图&am…

ios客户端学习笔记(五):学习Swift的关键字和容易弄混的符号

1. 关键字 下面是Swift语言中的常见关键字及其说明和代码应用实例&#xff1a; class&#xff1a;定义一个类&#xff0c;用于封装一组相关的属性和方法。 示例代码&#xff1a; class Person {var name: String ""var age: Int 0 }struct&#xff1a;定义一个…

网络安全与攻防-常见网络安全攻防

目录 攻击手段&防御策略 阻断服务攻击&#xff08;DoS&#xff09; 地址解析欺骗&#xff08;ARP攻击&#xff09;&#xff08;Address Resolution Protocol spoofing&#xff09; 跨站脚本攻击&#xff08;XSS&#xff09; SQL注入 跨站请求伪造&#xff08;csrf&am…

AlgoC++第二课:线性回归

目录 线性回归前言1. 定义2. 房价预测案例2.1 定义问题&#xff1a;2.2 MSE是如何被定义的&#xff1f;(如何推导的&#xff1f;)2.3 总结 3. 代码3.1 C实现3.1.1 数据读取和处理3.1.2 线性回归模型3.1.3 参数输出和预测3.1.4 完整示例代码3.1.5 C知识点 3.2 python实现3.2.1 房…

【移动端网页布局】移动端网页布局基础概念 ① ( 移动端浏览器 | 移动端屏幕分辨率 | 移动端网页调试方法 )

文章目录 一、移动端浏览器二、移动端屏幕分辨率三、移动端网页调试方法 一、移动端浏览器 移动端浏览器 比 PC 端浏览器发展要晚 , 使用的技术比较新 , 对 HTML5 CSS3 支持较好 , 常见的浏览器如下 : UC / QQ / Opera / Chrom / 360 / 百度 / 搜狗 / 猎豹 国内的浏览器 基本…

算法套路十一 ——回溯法之组合型回溯

算法套路十一 ——回溯法之组合型回溯 该节是在上一节回溯法之子集型回溯的基础上进行描写&#xff0c;组合型回溯会在子集型回溯的基础上判断所选子集是否符合组合要求&#xff0c; 故请首先阅读上一节算法套路十——回溯法之子集型回溯 算法示例&#xff1a;LeetCode77. 组合…

【C++ 十八】C++ map/ multimap容器

C map/ multimap 容器 文章目录 C map/ multimap 容器前言1 map 基本概念2 map 构造和赋值3 map 大小和交换4 map 插入和删除5 map 查找和统计6 map 容器排序 总结 前言 本文包含map基本概念、map构造和赋值、map大小和交换、map插入和删除、map查找和统计、map容器排序。 1 m…

使用Glib中测试框架对C代码进行单元测试

C项目的测试框架比较常见的是Google的gtest&#xff08;前文CMake项目使用ctestgtest进行单元测试有使用实例介绍gtest&#xff0c;感兴趣的读者可以去看看&#xff09;&#xff0c;也有一些其它框架&#xff0c;比如Boost中的测试框架。这些框架虽然也可以测试C代码&#xff0…

Vue 消息订阅与发布

消息订阅与发布&#xff0c;也可以实现任意组件之间的通信。 订阅者&#xff1a;就相当于是我们&#xff0c;用于接收数据。 发布者&#xff1a;就相当于是媒体&#xff0c;用于传递数据。 安装消息订阅与发布插件&#xff1a; 在原生 JS 中 不太容易实现消息订阅与发布&…

Unity-ML-Agents-代码解读-RollerBall

使用版本&#xff1a;https://github.com/Unity-Technologies/ml-agents/releases/tag/release_19 文件路径&#xff1a;ml-agents-release_19/docs/Learning-Environment-Create-New.md 20和19的在rollerBall上一样&#xff1a;https://github.com/Unity-Technologies/ml-ag…

CSDN博客编写教程

这里写自定义目录标题 欢迎使用Markdown编辑器新的改变功能快捷键合理的创建标题&#xff0c;有助于目录的生成如何改变文本的样式插入链接与图片如何插入一段漂亮的代码片生成一个适合你的列表创建一个表格设定内容居中、居左、居右SmartyPants 创建一个自定义列表如何创建一个…

osg widget 试用 方法

按钮 一个常见的 osg::Widget 就是按钮。下面的代码展示了如何使用 osg::Switch 和 osgText 创建一个简单的按钮&#xff1a; osg::ref_ptr<osg::Switch> buttonSwitch new osg::Switch(); osg::ref_ptr<osgText::Text> buttonText new osgText::Text(); buttonT…