[Algorithm][动态规划][子数组/子串问题][最大子数组和][环形子数组的最大和][乘积最大子数组][乘积为正数的最长子数组长度]详细讲解

本文主要是介绍[Algorithm][动态规划][子数组/子串问题][最大子数组和][环形子数组的最大和][乘积最大子数组][乘积为正数的最长子数组长度]详细讲解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

  • 1.最大子数组和
    • 1.题目链接
    • 2.算法原理详解
    • 3.代码实现
  • 2.环形子数组的最大和
    • 1.题目链接
    • 2.算法原理详解
    • 3.代码实现
  • 3.乘积最大子数组
    • 1.题目链接
    • 2.算法原理详解
    • 3.代码实现
  • 4.乘积为正数的最长子数组长度
    • 1.题目链接
    • 2.算法原理详解
    • 3.代码实现


1.最大子数组和

1.题目链接

  • 最大子数组和

2.算法原理详解

  • 思路
    • 确定状态表示 -> dp[i]的含义

      • i位置元素为结尾的所有子数组中的最大和
        请添加图片描述
    • 推导状态转移方程

      • dp[i] = max(nums[i], dp[i - 1] + nums[i])
        请添加图片描述
    • 初始化:dp表多开一个虚拟结点,避免处理边界

      • dp[0] = 0
    • 确定填表顺序:从左往右

    • 确定返回值:整个dp表里的最大值


3.代码实现

int maxSubArray(vector<int>& nums) 
{int n = nums.size();vector<int> dp(n + 1);dp[0] = 0;int ret = INT_MIN;for(int i = 1; i <= n; i++){dp[i] = max(nums[i - 1], dp[i - 1] + nums[i - 1]);ret = max(ret, dp[i]);}return ret;
}

2.环形子数组的最大和

1.题目链接

  • 环形子数组的最大和

2.算法原理详解

  • 本题因为有环形数组,问题并不好处理,但是可以将问题转化为最大子数组和

    • 若最大子数组和出现在中间,则与最大子数组和解法无异
    • 若最大子数组和出现在两边,则可以将问题转化为找中间的最小子数组和
    • 两个情况合并,找其中的最大和即可
      请添加图片描述
  • 思路

    • 确定状态表示 -> dp[i]的含义

      • f[i]:以i位置元素为结尾的所有子数组中的最大和
      • g[i]:以i位置元素为结尾的所有子数组中的最小和
        请添加图片描述
    • 推导状态转移方程

      • f[i] = max(nums[i], f[i - 1] + nums[i])
      • g[i] = min(nums[i], g[i - 1] + nums[i])
        请添加图片描述
    • 初始化:dp表多开一个虚拟结点,避免处理边界

      • f[0] = g[0] = 0 -> 确保dp[0]的值,不会影响后面的求和
    • 确定填表顺序:从左往右

    • 确定返回值:sum == gmin ? fmax : max(fmax, sum - gmin)


3.代码实现

int maxSubarraySumCircular(vector<int>& nums) 
{int n = nums.size();vector<int> f(n + 1);vector<int> g(n + 1);int fmax = INT_MIN, gmin = INT_MAX, sum = 0;for(int i = 1; i <= n; i++){int x = nums[i - 1];sum += x;f[i] = max(x, x + f[i - 1]);fmax = max(fmax, f[i]);g[i] = min(x, x + g[i - 1]);gmin = min(gmin, g[i]);}return sum == gmin ? fmax : max(fmax, sum - gmin);
}

3.乘积最大子数组

1.题目链接

  • 乘积最大子数组

2.算法原理详解

  • 思路
    • 确定状态表示 -> dp[i]的含义

      • f[i]:以i位置元素为结尾的所有子数组中的最大乘积
      • g[i]:以i位置元素为结尾的所有子数组中的最小乘积
        请添加图片描述
    • 推导状态转移方程

      • f[i] = max(nums[i], f[i - 1] * nums[i], g[i - 1] * nums[i])
      • g[i] = min(nums[i], f[i - 1] * nums[i], g[i - 1] * nums[i])
        请添加图片描述
    • 初始化:dp表多开一个虚拟结点,避免处理边界

      • f[0] = g[0] = 1 -> 确保dp[0]的值,不会影响后面的乘积
    • 确定填表顺序:从左往右

    • 确定返回值:f表里的最大值


3.代码实现

int maxProduct(vector<int>& nums) 
{int n = nums.size();vector<int> f(n + 1), g(n + 1);f[0] = g[0] = 1;int ret = INT_MIN;for(int i = 1; i <= n; i++){f[i] = max(nums[i - 1], max(f[i - 1] * nums[i - 1], g[i - 1] * nums[i - 1]));g[i] = min(nums[i - 1], min(f[i - 1] * nums[i - 1], g[i - 1] * nums[i - 1]));ret = max(ret, f[i]);}return ret;
}

4.乘积为正数的最长子数组长度

1.题目链接

  • 乘积为正数的最长子数组长度

2.算法原理详解

  • 思路
    • 确定状态表示 -> dp[i]的含义

      • f[i]:以i位置元素为结尾的所有子数组中,乘积为正数的最长长度
      • g[i]:以i位置元素为结尾的所有子数组中,乘积为负数的最长长度
        请添加图片描述
    • 推导状态转移方程
      请添加图片描述

    • 初始化:dp表多开一个虚拟结点,避免处理边界

      • f[0] = g[0] = 0
    • 确定填表顺序:从左往右,两个表一起填

    • 确定返回值:f表里的最大值


3.代码实现

int getMaxLen(vector<int>& nums) 
{int n = nums.size();vector<int> f(n + 1), g(n + 1);int ret = INT_MIN;for(int i = 1; i <= n; i++){if(nums[i - 1] > 0){f[i] = f[i - 1] + 1;g[i] = g[i - 1] == 0 ? 0 : g[i - 1] + 1;}else if(nums[i - 1] < 0){f[i] = g[i - 1] == 0 ? 0 : g[i - 1] + 1;g[i] = f[i - 1] + 1;}ret = max(ret, f[i]);}return ret;
}

这篇关于[Algorithm][动态规划][子数组/子串问题][最大子数组和][环形子数组的最大和][乘积最大子数组][乘积为正数的最长子数组长度]详细讲解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1012991

相关文章

Redis连接失败:客户端IP不在白名单中的问题分析与解决方案

《Redis连接失败:客户端IP不在白名单中的问题分析与解决方案》在现代分布式系统中,Redis作为一种高性能的内存数据库,被广泛应用于缓存、消息队列、会话存储等场景,然而,在实际使用过程中,我们可能... 目录一、问题背景二、错误分析1. 错误信息解读2. 根本原因三、解决方案1. 将客户端IP添加到Re

详谈redis跟数据库的数据同步问题

《详谈redis跟数据库的数据同步问题》文章讨论了在Redis和数据库数据一致性问题上的解决方案,主要比较了先更新Redis缓存再更新数据库和先更新数据库再更新Redis缓存两种方案,文章指出,删除R... 目录一、Redis 数据库数据一致性的解决方案1.1、更新Redis缓存、删除Redis缓存的区别二

oracle数据库索引失效的问题及解决

《oracle数据库索引失效的问题及解决》本文总结了在Oracle数据库中索引失效的一些常见场景,包括使用isnull、isnotnull、!=、、、函数处理、like前置%查询以及范围索引和等值索引... 目录oracle数据库索引失效问题场景环境索引失效情况及验证结论一结论二结论三结论四结论五总结ora

JAVA中整型数组、字符串数组、整型数和字符串 的创建与转换的方法

《JAVA中整型数组、字符串数组、整型数和字符串的创建与转换的方法》本文介绍了Java中字符串、字符数组和整型数组的创建方法,以及它们之间的转换方法,还详细讲解了字符串中的一些常用方法,如index... 目录一、字符串、字符数组和整型数组的创建1、字符串的创建方法1.1 通过引用字符数组来创建字符串1.2

element-ui下拉输入框+resetFields无法回显的问题解决

《element-ui下拉输入框+resetFields无法回显的问题解决》本文主要介绍了在使用ElementUI的下拉输入框时,点击重置按钮后输入框无法回显数据的问题,具有一定的参考价值,感兴趣的... 目录描述原因问题重现解决方案方法一方法二总结描述第一次进入页面,不做任何操作,点击重置按钮,再进行下

最新版IDEA配置 Tomcat的详细过程

《最新版IDEA配置Tomcat的详细过程》本文介绍如何在IDEA中配置Tomcat服务器,并创建Web项目,首先检查Tomcat是否安装完成,然后在IDEA中创建Web项目并添加Web结构,接着,... 目录配置tomcat第一步,先给项目添加Web结构查看端口号配置tomcat    先检查自己的to

使用Nginx来共享文件的详细教程

《使用Nginx来共享文件的详细教程》有时我们想共享电脑上的某些文件,一个比较方便的做法是,开一个HTTP服务,指向文件所在的目录,这次我们用nginx来实现这个需求,本文将通过代码示例一步步教你使用... 在本教程中,我们将向您展示如何使用开源 Web 服务器 Nginx 设置文件共享服务器步骤 0 —

解决mybatis-plus-boot-starter与mybatis-spring-boot-starter的错误问题

《解决mybatis-plus-boot-starter与mybatis-spring-boot-starter的错误问题》本文主要讲述了在使用MyBatis和MyBatis-Plus时遇到的绑定异常... 目录myBATis-plus-boot-starpythonter与mybatis-spring-b

SpringBoot集成SOL链的详细过程

《SpringBoot集成SOL链的详细过程》Solanaj是一个用于与Solana区块链交互的Java库,它为Java开发者提供了一套功能丰富的API,使得在Java环境中可以轻松构建与Solana... 目录一、什么是solanaj?二、Pom依赖三、主要类3.1 RpcClient3.2 Public

手把手教你idea中创建一个javaweb(webapp)项目详细图文教程

《手把手教你idea中创建一个javaweb(webapp)项目详细图文教程》:本文主要介绍如何使用IntelliJIDEA创建一个Maven项目,并配置Tomcat服务器进行运行,过程包括创建... 1.启动idea2.创建项目模板点击项目-新建项目-选择maven,显示如下页面输入项目名称,选择