想要精通算法和SQL的成长之路 - 环形子数组的最大和

2023-09-18 21:10:04

想要精通算法和SQL的成长之路 - 环形子数组的最大和

前言

想要精通算法和SQL的成长之路 - 系列导航

一. 环形子数组的最大和

原题链接

在这里插入图片描述

在写这道题目之前,可以先看下这个题:最大子数组和 。本题是它的进阶版本,在原本的基础上,有一个环状的数组。那么我们如果将其平铺开来,就是一个两段数组拼接而成。

我们假设我们的数组是这样的:[ x, x, x, x, x, x, x, x, x, x, x, x, x, x, x, x] 。那么环形子数组的最大和,这个数组必定只有两种情况:

  • 在数组内部:[ x, x, x, x, x, x, x, x, x, x, x, x, x, x, x, x]
  • 由数组头和数组的尾部连结而成:[ x, x, x, x, x, x, x, x, x, x, x, x, x, x, x, x]

如果是前者,那么看 最大子数组和 的题解即可:

public int maxSubArray(int[] nums) {
    // 初始化dp数组
    int[] dp = new int[nums.length];
    dp[0] = nums[0];
    int res = nums[0];
    // 遍历数组,dp[0]就是第一个数字它本身
    for (int i = 1; i < nums.length; i++) {
        dp[i] = Math.max(dp[i - 1] + nums[i], nums[i]);
        res = Math.max(res, dp[i]);
    }
    return res;
}

如果是后者,我们有以下分析:[ x, x, x, x, x, x, x, x, x, x, x, x, x, x, x, x]

  • 左侧红色区域中:最右侧的数值必定是正数。 因为如果它是负数,那么它对于最大值的影响必定是负影响,不可能计入结果当中。同理,紧跟右侧的相邻数必定是负数。如果是正数,那么他也应该计入到结果中。
  • 同理,右侧红色区域中:最左侧的数值必定是正数。 左侧相邻的数必定是负数。
  • 即:[ x, x, x, 正数, 负数, x, x, x, x, x, x, x, x, 负数, 正数, x]

那么如果红色区域是:最大子数组和。中间部分就是:最小子数组和。

最大子数组和 = 数组总和 - 最小子数组和。这部分代码就是:

int min = dp[0];
// 这种情况我们视为左右两端成环的情况,那么求最小和数组的时候,必定不包含左右两侧端点
for (int i = 1; i < len - 1; i++) {
    dp[i] = nums[i] + Math.min(0, dp[i - 1]);
    min = Math.min(min, dp[i]);
}

我们再加一个变量,用来存储数组总和即可,完整代码:

public int maxSubarraySumCircular(int[] nums) {
    int len = nums.length;
    int[] dp = new int[len];
    dp[0] = nums[0];
    int res = dp[0];
    int sum = dp[0];
    for (int i = 1; i < len; i++) {
        dp[i] = Math.max(dp[i - 1] + nums[i], nums[i]);
        res = Math.max(res, dp[i]);
        sum += nums[i];
    }

    int min = dp[0];
    for (int i = 1; i < len; i++) {
        dp[i] = nums[i] + Math.min(0, dp[i - 1]);
        min = Math.min(min, dp[i]);
    }
    return Math.max(sum - min, res);
}

1.1 空间优化

先说下求最大值部分,参考:最大子数组和

int max = nums[0], pre = 0;
for (int num : nums) {
    pre = Math.max(pre + num, num);
    max = Math.max(max, pre);
}

也就是说,当我们发现dp整个运算过程中,只用到了两种变量:dp[i]dp[i-1]的时候,我们就可以用变量去替换整个数组。
那么求最小值部分同理:

int min = nums[0], pre2 = 0;
for (int num : nums) {
    pre2 = Math.min(pre2 + num, num);
    min = Math.min(min, pre2);
}

两个数组整合起来,再加一个变量sum代表数组总和,结果就出来了:

public int maxSubarraySumCircular(int[] nums) {
    int max = nums[0], pre = 0, min = nums[0], pre2 = 0, sum = 0;
    for (int num : nums) {
        // 求最大值和
        pre = Math.max(pre + num, num);
        max = Math.max(max, pre);
        // 求最小值和
        pre2 = Math.min(pre2 + num, num);
        min = Math.min(min, pre2);
        sum += num;
    }
    if (max < 0) {
        return max;
    }
    return Math.max(max, sum - min);
}
更多推荐

【网络协议】Http-下

因为Http是无状态的,所以为了协助Web保持状态,Cookie诞生了。下面中是百度百科关于Cookie和Session的解释:Cookie:举例来说,一个Web站点可能会为每一个访问者产生一个唯一的ID,然后以Cookie文件的形式保存在每个用户的机器上。如果使用浏览器访问Web,会看到所有保存在硬盘上的Cookie

【Git】03-GitHub

文章目录1.GitHub核心功能2.GitHub搜索项目3.GitHub搭建个人博客4.团队项目创建5.git工作流选择5.1需要考虑的因素5.2主干开发5.2GitFlow5.3GitHubFlow5.4GitLabFlow(带生产分支)5.4GitLabFlow(带环境分支)5.4GitLabFlow(带发布分支)

Ceph入门到静态-deep scrub 深度清理处理

9.6洗刷REPORTDOCUMENTATIONBUG#除了为对象创建多个副本外,Ceph还可通过洗刷归置组来确保数据完整性(请参见第1.3.2节“归置组”了解有关归置组的详细信息)。Ceph的洗刷类似于在对象存储层运行fsck。对于每个归置组,Ceph都会生成一个包含所有对象的编目,并比较每个主对象及其副本,以确保不

如何解决 Spring Boot Actuator 的未授权访问漏洞

SpringBootActuator的作用是提供了一组管理和监控端点,允许你查看应用程序的运行时信息,例如健康状态、应用程序信息、性能指标等。这些端点对于开发、测试和运维团队来说都非常有用,可以帮助快速诊断问题、监控应用程序的性能,并采取必要的措施来维护和管理应用程序。SpringBootActuator未授权访问的配

LeetCode 40. Combination Sum II【回溯,剪枝】中等

本文属于「征服LeetCode」系列文章之一,这一系列正式开始于2021/08/12。由于LeetCode上部分题目有锁,本系列将至少持续到刷完所有无锁题之日为止;由于LeetCode还在不断地创建新题,本系列的终止日期可能是永远。在这一系列刷题文章中,我不仅会讲解多种解题思路及其优化,还会用多种编程语言实现题解,涉及

LeetCode 39. Combination Sum【回溯,剪枝】中等

本文属于「征服LeetCode」系列文章之一,这一系列正式开始于2021/08/12。由于LeetCode上部分题目有锁,本系列将至少持续到刷完所有无锁题之日为止;由于LeetCode还在不断地创建新题,本系列的终止日期可能是永远。在这一系列刷题文章中,我不仅会讲解多种解题思路及其优化,还会用多种编程语言实现题解,涉及

Android.bp常用语法和预定义属性

介绍Android.bp是Android构建系统中用于定义模块和构建规则的配置文件,它使用一种简单的声明式语法。以下是Android.bp的一些常见语法规则和约定:注释:单行注释使用//符号。多行注释使用/和/包围。和go语言相同//这是单行注释/*这是多行注释*/模块定义:每个模块都以module_type字段开始,

hadoop集群搭建

vim/etc/hosts192.168.1.2Master.Hadoop192.168.1.3Slave1.Hadoop192.168.1.4Slave2.Hadoop192.168.1.5Slave3.Hadoop若能用主机名进行ping通,说明刚才添加的内容,在局域网内能进行DNS解析。hadoop:https:

【iOS】浅析static,const,extern关键字

文章目录前言一、staticstatic修饰局部变量static修饰全局变量总结二、const三、extern声明全局变量声明函数在头文件中使用总结前言笔者本周在学习单例模式时,用到了static关键字,特此总结博客记录学习static,const,extern关键字的过程一、staticstatic——静态,我们将用

【golang】实现通用的get/post请求(接受一个 URL 和一个结构体参数)

通用的GET请求实现一个通用的GET请求函数,该函数接受一个URL和一个结构体参数,并将结构体参数编码为查询参数。以下是一个通用的示例代码:packagemainimport("fmt""net/http""net/url""reflect""strings")funcgetFunc(baseUrlstring,str

Learn Prompt-ChatGPT 精选案例:简单介绍

恭喜你!现在你已经学会了如何编写提示语。本节主要讨论的是如何使用提示语来解决我们在日常或工作中遇到的任务。如果你已经有了一个提示语集,如何决定哪些提示语适合手头的任务?在本节中,我们将通过一些实际例子来给你提供灵感。在挑选案例的时候,我们更加希望展示的是如何将复杂的工作任务拆解成相互关联的小任务。例如在PPT制作中,我

热文推荐