题目

洛谷:P1115 最大子段和

LeetCode:最大子序和

给出一个长度为 \(n\) 的序列 \(a\),选出其中连续且非空的一段使得这段和最大。

挺经典的一道题目,下面分别介绍 \(O(n)\) 的 DP 做法、前缀和做法,以及 \(O(n\log n)\) 的分治做法。

DP 做法

用 \(d_i\) 表示结尾为位置 \(i\) 的最大区间和,则有

\[d_i=\max(d_{i-1},0)+a_i
\]

问题的答案即为 \(\max\{d_i \mid i\in[1,n]\}\)。

编写代码时不需要开 \(d\) 数组,用变量 last_d 记录 \(d_{i-1}\),变量 ans 记录 \(\max\{d_i\}\),并在扫描时动态更新即可。

时间复杂度 \(O(n)\),空间复杂度 \(O(1)\)。

核心代码如下:

maxn = int(2e5 + 5)
arr = [0 for _ in range(maxn)] # 从下标 1 开始存 # 输入过程略…… ans = None
last_d = 0
for i in range(1, n + 1):
temp_ans = max(last_d, 0) + arr[i]
if ans is None or temp_ans > ans:
ans = temp_ans
last_d = temp_ans
print(ans)

前缀和做法

将数列前 \(n\) 项的和记为 \(sum_n\):

\[sum_n=\sum_{i=1}^n a_i
\]

可以用前缀和快速求区间和:

\[\sum_{i=x}^y a_i = sum_y - sum_{x-1}
\]

用 \(d_i\) 表示结尾为位置 \(i\) 的最大区间和,则有

\[d_i=sum_i-\min\{sum_j \mid j<i\}
\]

问题的答案即为 \(\max\{d_i \mid i \in [1,n]\}\)。

编写代码时只需要开前缀和数组,无需开 \(d\) 数组,用变量 cur_min_pre_sum 记录 \(\min\{sum_j\}\),变量 ans 记录 \(\max\{d_i\}\),并动态维护即可。

时间复杂度 \(O(n)\),空间复杂度 \(O(n)\)。

核心代码如下:

maxn = int(2e5 + 5)
arr = [0 for _ in range(maxn)] # 原数组,从下标 1 开始存
pre_sum = [0 for _ in range(maxn)] # 前缀和数组 # 输入过程略…… # 预处理前缀和
for i in range(1, n + 1):
pre_sum[i] = pre_sum[i - 1] + arr[i] cur_min_pre_sum = 0
ans = None
for i in range(1, n + 1):
temp_ans = pre_sum[i] - cur_min_pre_sum
if ans is None or temp_ans > ans:
ans = temp_ans
cur_min_pre_sum = min(cur_min_pre_sum, pre_sum[i])
print(ans)

分治做法

若有一区间 \([start,stop)\),区间中点为 \(mid\),其最大子段和对应的子区间为 \([i,j)\),则 \([i,j)\) 只有以下三种情况:

  1. \([i,j)\) 完全在左子区间 \([start,mid)\) 内;
  2. \([i,j)\) 完全在右子区间 \([mid,stop)\) 内;
  3. \([i,j)\) 横跨中点 \(mid\)。

求出这三种情况下的值,取最大的即可。

前两种情况可通过递归求解,求解第三种情况需要一点技巧,方法是从中点出发分别向左右两边延伸

时间复杂度 \(O(n\log n)\)。

核心代码如下:

maxn = int(2e5 + 5)
arr = [0 for _ in range(maxn)] # 从下标 1 开始存 # 从位置 mid - 1 开始向左延伸的最大区间和
# 注:左子区间 [start, mid)
def mid_lmax(start: int, mid: int) -> int:
ans = None
cur_sum = 0
for i in range(mid - 1, start - 1, -1):
cur_sum += arr[i]
if ans is None or cur_sum > ans:
ans = cur_sum
return ans # 从位置 mid 开始向右延伸的最大区间和
# 注:右子区间 [mid, stop)
def mid_rmax(mid: int, stop: int) -> int:
ans = None
cur_sum = 0
for i in range(mid, stop):
cur_sum += arr[i]
if ans is None or cur_sum > ans:
ans = cur_sum
return ans # [start, stop) 的最大子区间和
def solve(start: int, stop: int) -> int:
if stop - start == 1:
return arr[start]
mid = (start + stop) // 2
only_lmax = solve(start, mid) # 完全在左子区间内
only_rmax = solve(mid, stop) # 完全在右子区间内
span_max = mid_lmax(start, mid) + mid_rmax(mid, stop) # 横跨中点
return max(only_lmax, only_rmax, span_max)

三种方法求解最大子区间和:DP、前缀和、分治的更多相关文章

  1. 使用三种方法求解前N个正整数的排列

    本篇博文给大家介绍前N个正整数的排列求解的三种方式.第一种是暴力求解法:第二种则另外声明了一个长度为N的数组,并且将已经排列过的数字保存其中:第三种方式则采用了另外一种思路,即首先获取N个整数的升序排 ...

  2. Java/JSP获得客户端网卡MAC地址的三种方法解析

    java/jsp获得客户端(IE)网卡MAC地址的方法大概有三种. 1.通过命令方式,在客户端执行Ipconfig 等等.(java/jsp) 2.通过ActiveX的方法.(jsp) 3.通过向13 ...

  3. 数组k平移三种方法(java)

    上代码,本文用了三种方法实现,时间复杂度不一样,空间复杂度都是o(1): public class ArrayKMove { /** * 问题:数组的向左k平移,k小于数组长度 * @param ar ...

  4. C#中??和?分别是什么意思? 在ASP.NET开发中一些单词的标准缩写 C#SESSION丢失问题的解决办法 在C#中INTERFACE与ABSTRACT CLASS的区别 SQL命令语句小技巧 JQUERY判断CHECKBOX是否选中三种方法 JS中!=、==、!==、===的用法和区别 在对象比较中,对象相等和对象一致分别指的是什么?

    C#中??和?分别是什么意思? 在C#中??和?分别是什么意思? 1. 可空类型修饰符(?):引用类型可以使用空引用表示一个不存在的值,而值类型通常不能表示为空.例如:string str=null; ...

  5. 像画笔一样慢慢画出Path的三种方法(补充第四种)

    今天大家在群里大家非常热闹的讨论像画笔一样慢慢画出Path的这种效果该如何实现. 北京-LGL 博客号@ligl007发起了这个话题.然后各路高手踊跃发表意见.最后雷叔 上海-雷蒙 博客号@雷蒙之星 ...

  6. JAVA之线程同步的三种方法

    最近接触到一个图片加载的项目,其中有声明到的线程池等资源需要在系统中线程共享,所以就去研究了一下线程同步的知识,总结了三种常用的线程同步的方法,特来与大家分享一下.这三种方法分别是:synchroni ...

  7. java解析xml的三种方法

    java解析XML的三种方法 1.SAX事件解析 package com.wzh.sax; import org.xml.sax.Attributes; import org.xml.sax.SAXE ...

  8. 【Android】Eclipse自动编译NDK/JNI的三种方法

    [Android]Eclipse自动编译NDK/JNI的三种方法 SkySeraph Sep. 18th  2014 Email:skyseraph00@163.com 更多精彩请直接访问SkySer ...

  9. DataTable数据批量写入数据库三种方法比较

    DataTable数据批量写入数据库三种方法比较 标签: it 分类: C#1)   insert循环插入:2)   sqldataadapter.update(dataset,tablename); ...

随机推荐

  1. 解决servlet中get方式中中文乱码问题前驱(一):装饰者模式再理解

    package day02; import java.io.BufferedReader; import java.io.FileReader; import java.io.IOException; ...

  2. reids rdb与aof

    rdb:时合高并发场景,容易备份恢复,会丢失部分数据 1.默认开启的方式,可以进过压缩,可以根据时间点生成快照 2.数据量大的情况下恢复快 3.bgsave一边开启fork保存文件,一边继续响应客户端 ...

  3. Kubernetes集群部署笔记

    本作品由Galen Suen采用知识共享署名-非商业性使用-禁止演绎 4.0 国际许可协议进行许可.由原作者转载自个人站点. 概述 本文用于整理基于Debian操作系统使用kubeadm工具部署Kub ...

  4. Ajax重构

    Ajax重构简介 Ajax的实现主要依赖于XMLHttpRequest对象,但是在调用其进行异步数据传输时,由于XMLHttpRequest对象的实例在处理事件完成后就会被销毁,所以如果不对该对象进行 ...

  5. Linux学习笔记 - Linux快捷操作及常用命令

    一.快捷键 剪切光标前的内容 Ctrl + u 剪切光标至行末的内容 Ctrl + k 粘贴 Ctrl + u 或 Ctrl +k 的内容 Ctrl + y 移动光标到行末 Ctrl + e 移动光标 ...

  6. Java基础(一)——面向对象

    一.对象 1.成员变量和局部变量的区别 两类变量同名时,局部变量具有更高的优先级. 作用域不同:局部变量的作用域仅限于定义它的方法,作用于函数或者语句中:成员变量的作用域在整个类中. 初始值不同:Ja ...

  7. 各色Tarjan集合

    #include<bits/stdc++.h> using namespace std; const int N=100000,M=200000; //所有Tarjan都要: // dfn ...

  8. www迁移

    www迁移主要就是2部分: 1)官网页面架构,即 ./drupal/index.php 2)官网图片,即 ./drupal/assets/ 目录下的文件 1. 在ubuntu上搭建基础v1.0环境 2 ...

  9. Input 只能输入数字,数字和字母等的正则表达式

    JS只能输入数字,数字和字母等的正则表达式 1.文本框只能输入数字代码(小数点也不能输入) <input onkeyup="this.value=this.value.replace( ...

  10. 云真机兼容性自动化工具测试解决方案_www.alltesting.cn

    问题和背景 不同类型的品牌和硬件环境.不同版本的android操作系统.IO操作系统,以及不同的分辨率,造成相同的APP在不同的设备可能存在缺陷. 兼容性测试,就是让APP.小程序.H5程序,在所有的 ...