题目

洛谷: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. Linux centos 安装 tomcat 7

    一.tomcat 下载 1.官网下载 如下图点击第1个步骤,第2个步骤右键复制链接地址 # 到linux 下执行下载命令 wget http://mirrors.shu.edu.cn/apache/t ...

  2. MutationObserver API

    1.概述 MutationObserver接口提供了监视对DOM树所做更改的能力.它被设计为旧的Mutation Events功能的替代品,该功能是DOM3 Events规范的一部分. 但是,它与Mu ...

  3. Servlet学习笔记(三)之HttpServletRequest

    HttpServletRequest(HttpServletRequest 想比 ServletRequest 添加与协议相关 API)对象代表客户端的请求,当客户端通过HTTP协议访问服务器时,HT ...

  4. 从IT圈“鄙视链”看前端开发有多难?

    如今"鄙视链"体现在生活的方方面面,各行各业都有默认一致的鄙视链.IT圈子因为开发语言多样.工程师岗位种类多.技术框架多,也有自己圈子内的鄙视链.按照开发工程师的岗位形成的鄙视链是 ...

  5. Ordering the Soldiers 题解

    CodeChef:ORDERS 简化题意: \(n\) 个人排队,给定每个人需要向左移动几个,求最终排列. 即还原逆序对. 错误想法 既然知道每个人向左移动 \(a_i\) 个,那就相当于让他的排名 ...

  6. IP头详解

    IP包头长度(Header Length):长度4比特.这个字段的作用是为了描述IP包头的长度,因为在IP包头中有变长的可选部分.该部分占4个bit位,单位为32bit(4个字节),即本区域值= IP ...

  7. matlab纹理映射之地球

    %地球 cla reset; load topo; [x,y,z] = sphere(45); s = surface(x,y,z,'facecolor','texturemap','cdata',t ...

  8. Abp Vnext3 vue-admin-template(三获取用户信息)

    因为获取用户比较简单,只需要把用户名及头像地址赋值即可(也许理解错误,如果发现请告知谢谢), 首先将src\api\usr.js中的url请求地址改为以下代码 export function getI ...

  9. IDEA创建Maven项目做Java Web时无WEB-INF/classes的问题

    昨天开始学习Java Web中的Servlet,学到用IntelliJ IDEA创建Java Web项目时,跟着课程上老师的步骤一步步做,却发现运行时Servlet找不到.坑爹的是,练习建项目时,一模 ...

  10. Spring Cloud Gateway 学习+实践

    官网上给出的Spring Cloud Gateway特性如下图所示: 翻译过来就是: 基于 Spring Framework 5 ,Project Reactor 以及 Spring Boot 2.0 ...