题目

洛谷: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. -e $request_filename + nginx内置变量

    -e表示只要filename存在,则为真,不管filename是什么类型,当然这里加了!就取反额外的一些-e filename 如果 filename存在,则为真-d filename 如果 file ...

  2. rtvue-lowcode:一款基于uniapp框架和uview组件库的开源低代码开发平台

    rtvue-lowcode低代码开发平台 rtvue-lowcode一款基于uniapp框架和uview组件库的低代码开发平台,项目提供可视化拖拽编辑器,采用MIT开源协议,适用于app.小程序等项目 ...

  3. java js转码

    public static String escape(String src) { int i; char j; StringBuffer tmp = new StringBuffer(); tmp. ...

  4. 将 VS2017下开发的程序, 部署到其他电脑上运行

    关键步骤:设置Release,如下图 如果无法直接执行,则安装ALI213-Microsoft.Visual.C++.2017.Redistributable.Package.x86.x64

  5. linux centos7 模拟垃圾回收站功能以及 crontab 定时任务的设置

    2021-08-04 1. 安装 环境:CentOS Linux release 7.5.1804 (Core) # 将 saferm.sh 拷贝到 /bin 目录下面 git clone git:/ ...

  6. 手把手教你在 SuperEdge 上用 EdgeX Foundry 接入 IoT 设备

    作者 连泓乔,华南理工计算机科学与技术大三在读,主要研究容器领域,Kubernetes.容器等云原生技术爱好者,SuperEdge 优秀贡献者. 王冬,腾讯云研发工程师,专注于 Kubernetes. ...

  7. linux清空文件

    https://www.cnblogs.com/mrwang1101/p/6166326.html

  8. 以人为本打造“超职季”IP,58同城精准匹配企业招聘与打工人

    撰文 |懂懂 编辑 | 秦言 来源:懂懂笔记 在大手笔培育IP的背后,58同城是如何考量的? 在餐厅当服务员的李阿姨今年54岁了.她的女儿马上研究生毕业,非常喜欢陈伟霆,手机屏保都是他.李阿姨没想到, ...

  9. python模块--__future__(向上兼容模块)

    py2.7   unicode_literals 将字符串默认视为unicode, 即u'xxx'和'xxx'将是一样的, 而再想表示字节需用b'xxx'表示 division / 将表示正常除法操作 ...

  10. 日期SQL 脚本

    一个月第一天的 SELECT DATEADD(mm, DATEDIFF(mm,0,getdate()), 0)本周的星期一     SELECT DATEADD(wk, DATEDIFF(wk,0,g ...