最大子矩阵:前缀和、动态规划

最近在学习动态规划,在牛客上刷题时碰到了这一题。其实最初的想法是暴力和前缀和,但是时间复杂度极高,需要套4层循环。后来去网上搜了一下相关的题解和做法,进而了解到了前缀和+线性动态规划的做法。但是在成功做出这题之前,个人感觉所搜到的博主讲解偏向于代码的编写,对于我这种初入算法的小白来说还是蛮费力气的,所以本节内容我将和大家一起从算法原理到代码一一剖析,争取写出清晰且容易理解的算法思路。

题目链接:最大子矩阵_牛客题霸_牛客网 (nowcoder.com)

 

首先我们来明确一下题目要求:本题要求的是我们在一个给定的N x N 的“矩阵”中找到 “元素和”最大的“子矩阵”。这个N x N 的矩阵其实就是一个二维数组,我们把它理解为一个矩阵。那么它的子矩阵就是矩阵中的某一片矩阵型的区域,那所要求的“最大子矩阵”的自然是找到在所有子矩阵中,矩阵元素和最大的那个矩阵。

我们先来看示例1来帮助我们理解一下题意:

我们可以发现,除该子矩阵外,在其他任意地方随意圈出一个子矩阵中的元素和均比图示矩阵的元素和小,此时我们就找到了该矩阵的最大子矩阵。

以下是两种解法:

 

 

说实话,第一种纯暴力解法就不用看了,数据量稍微大一点,就超时了。

第二种使用二维数组前缀和对原始数组进行了预处理,得到了二维前缀和数组,之后在我们求圈定范围的子矩阵元素和时会带来不小便利,但是4层循环O(n^4)的时间复杂度仍然不可小觑。 

那么我们有什么方法能减少时间复杂度呢?

我们先来了解一维数组的前缀和:

一维数组的前缀和是指将数组中从起始位置到当前位置的所有元素相加的结果,即上图中的prefix数组。

我们再来复习一下求一维数组最大子序列所使用到的线性dp算法:

算法原理:

状态表示:dp[i]所表示的是以i结尾的所有子数组中元素的最大和。 

状态转移方程:用来更新动态规划数组dp[i]:

  • nums[i - 1]表示当前位置i对应的原始数组 nums 中的元素值。

  • dp[i - 1] + nums[i - 1] 表示从当前位置向前延伸的子序列的和,即以 nums[i - 1] 结尾的子序列和加上当前位置的元素值。这个值表示了当前位置开始的新子序列的和。

  • max(nums[i - 1], dp[i - 1] + nums[i - 1]) 选择了两种情况中的较大值:第一种情况是只包含当前元素 nums[i - 1],即以当前位置 i 结尾的子序列。第二种情况是将当前位置的元素加入到之前的子序列中,即从当前位置开始新的子序列。

  • 将较大值更新到 dp[i],表示以 nums[i - 1] 结尾的最大子序列和。

这样,通过动态规划数组 dp 的更新,每个位置 i 都计算出了以该位置结尾的最大子序列和,最终找到整个数组的最大子序列和。

由于最后要输出的是dp数组中的最大值,我们可以使用一个变量来记录以 i 位置为结尾的最大子序列和,这样空间复杂度就从O(N)减少到了O(1)。

 其实,对于我们这一题,也可以使用一维数组前缀和与线性dp来解决,我们只需要将二维数组“压缩”为一维数组就好了。那么压缩方式自然是使用前缀和了。

怎么压缩呢?压缩后如何使用前缀和搭配线性dp解题呢?我们接着往下看:

进行完这一步操作后,第 i 行中第 j 列的元素即为:第 j 列从起始位置到当前位置 i 的所有元素相加的结果。 

其实通过上面的例子,我们不难发现进行预处理后的前缀和数组,仍然可以表示原数组的元素,更方便的是对于求原数组圈定矩阵元素和,在处理后的数组中仅通过使用末行数组元素减去起始行前一行的数组元素就可以得到所求矩阵的元素和。

通过不同的末行对起始行的减法操作,我们最后可以得到如下序列:

 通过前后两张图的解析,其实我们不难发现在最终生成的任意一个子序列中,随机取一段连续的数字即可表示原数组中的子矩阵。即原矩阵中所有的子矩阵均可由生成的子序列得到。

大家一定要好好理解并验证上面的两张图和两段话,当理解通透了才便于进行后续代码的编写。

既然 原矩阵中所有的子矩阵均可由生成的子序列得到,那么最大子矩阵必定是所有子序列数组中的最大连续子序列(注意:是上面通过两层循环得到的10个子序列数组中最大连续子序列元素和)。

那么我们接下来的目标很简单,就是对10个子序列数组中的每一个进行对最大连续子序列元素和的求解。所以我们需要再加上一层循环,用来遍历子序列数组。并且我们需要一个变量 ans 来记录每个子序列数组中的最大子序列元素和,需要注意的是 ans 在每次进入循环之前要更新为0,防止对后续最大子序列的求解造成影响。同时需要一个变量maxi 来记录所有子序列数组中最大的连续子序列元素和,而这个值就是我们的最大子矩阵元素和。 

最终代码:

通常我们会对二维数组多申请一行和一列的空间并初始化为0,是为了dp的状态转移方程在使用时不需要对边界情况进行特殊处理,并且不对dp数组元素的结构造成影响,提高代码的简洁性。

#include <iostream>
#include <vector>
using namespace std;const int N = 101; // 数组的最大大小int main() {int n = 0;cin >> n; // 初始化大小为 (n+1) x (n+1) 的二维数组,所有元素为 0vector<vector<int> > arr(n + 1, vector<int>(n + 1, 0)); // 初始化大小为 (n+1) x (n+1) 的二维向量 dp,所有元素为 0vector<vector<int> > dp(n + 1, vector<int>(n + 1, 0)); // 输入矩阵的元素,并计算每列的前缀和for(int i = 1; i <= n; i++) {for(int j = 1; j <= n; j++) {cin >> arr[i][j];dp[i][j] = dp[i - 1][j] + arr[i][j]; // 计算每列的前缀和}}int maxi = -128 * 1E4; // 初始化最大值为一个该题中最小的数int ans = 0;// 遍历所有的行for(int i = 0; i < n; i++)  // 从第 0 行到第 n-1 行{for(int j = i + 1; j <= n; j++) // 从第 1 行到第 n 行{ ans = 0;// 遍历每一列,并计算当前行对的最大子序列和for(int k = 1; k <= n; k++)  // 遍历每一列的元素{// 计算当前行对的最大子序列和,并更新 ansans = max(dp[j][k] - dp[i][k], ans + dp[j][k] - dp[i][k]);// 更新目前为止找到的最大子序列和(maxi)maxi = max(ans, maxi);}}}cout << maxi << endl; // 输出最大子序列和return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.xdnf.cn/news/1421068.html

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

排除对象属性序列化的三种方式

说明&#xff1a;在项目里&#xff0c;经常可以看到以下日志内容&#xff0c;将对象序列化后直接打印出来&#xff0c;观察对象数据&#xff0c;判断当前处理逻辑正确与否。 &#xff08;以下信息来自&#xff1a;https://www.tl.beer/randbankcard.html生成器&#xff0c;信息…

python跟C++选哪个?

选择使用Python还是C取决于你的具体需求和项目背景。我这里有一套编程入门教程&#xff0c;不仅包含了详细的视频讲解&#xff0c;项目实战。如果你渴望学习编程&#xff0c;不妨点个关注&#xff0c;给个评论222&#xff0c;私信22&#xff0c;我在后台发给你。 在通信工程行业…

第六节笔记及作业----Lagent AgentLego 智能体应用搭建

关于 Agent 的相关理论 大语言模型存在一些局限性&#xff0c;比如会出现幻觉问题、有时效性问题以及可靠性问题。智能体的定义是具备感知、决策和行动能力的实体。智能体主要由感知部分、大脑部分和动作部分组成。智能体有多种类型&#xff0c;如 ReAct 类型&#xff08;侧重…

TCP服务器实现将客服端发送的信息广播发送(使用内核链表管理客户端信息)

目录 1.服务器端实现思路 2.服务器端代码 3.客户端代码 4.内核链表代码 5.运行格式 一、服务器端 二、客户端 6.效果 1.服务器端实现思路 Tcp广播服务初始化 等待客户端连接 广播发送 2.服务器端代码 #include "list.h" #include <signal.h> #def…

视频打赏系统源码

地球号&#xff1a;xiaobao0214520(WX) 支付对接&#xff0c;盒子推广&#xff0c;域名防封&#xff0c;等一系列功能皆为正常&#xff0c;后台账号密码:地球号&#xff1a;xiaobao0214520(WX)&#xff0c;测试网站,可以定制哦支付对接&#xff0c;盒子推广&#xff0c;域名防…

免费思维13招之七:空间型思维

免费思维13招之七:空间型思维 本篇给你带来的是空间型思维。 空间型思维,具体分为内部空间型思维和外部空间型思维。 什么叫内部空间型思维呢? 内部空间型就是充分利用现有空间或资源为社会提供免费服务,积累人气,增加流量,从而带动消费。 为什么你生意不好?为什么你…

python数据分析——matplotlib可视化基础

参考资料&#xff1a;活用pandas库 # 导入库 import pandas as pd import matplotlib.pyplot as plt # 导入数据 anscombepd.read_csv(r"...\seaborn常用数据案例\anscombe.csv") anscombe.head() 大多数基本图表的名字以plt.plot开头。 # 创建数据子集 # 只包含数…

图片转word如何转换?

要将图片转换为Word文档&#xff0c;你可以使用以下方法之一&#xff1a; 以上这些方法都可以帮助你将图片中的文本转换为可编辑的Word文档&#xff0c;你可以根据自己的喜好和需求选择其中一种方法来操作。 使用OCR软件或在线工具&#xff1a;有许多OCR&#xff08;Optical Ch…

【Spring】验证 @ServerEndpoint 的类成员变量线程安全

文章目录 前言猜想来源验证方法Controller 的情况ServerEndpoint 的情况 后记 前言 最近有 websocket 的需求。探索 ServerEndpoint 的类成员变量特点。 这里类比 Controller 讨论 ServerEndpoint 类成员变量是否线程安全。 猜想来源 网上的教程大多数都这么展示程序&#…

Ardupilot开源代码之Rover上路 - 后续1

Ardupilot开源代码之Rover上路 - 后续1 1. 源由2. 问题汇总2.1 问题1&#xff1a;飞控选择2.2 问题2&#xff1a;飞控安装位置和固定2.3 问题3&#xff1a;各种插头、插座配套2.4 问题4&#xff1a;分电板缺陷2.5 问题5&#xff1a;电机编码器接线及正反向问题2.6 问题6&#x…

树莓派python开发

树莓派自带thonny 点亮LED灯 import RPi.GPIO as GPIO import time# 设置GPIO模式为BCM GPIO.setmode(GPIO.BCM)# 设置LED引脚 led_pin 18# 设置LED引脚为输出 GPIO.setup(led_pin, GPIO.OUT)# 点亮LED GPIO.output(led_pin, GPIO.HIGH)# 延时2秒 time.sleep(2)# 关闭LED GPI…

电商核心技术揭秘55:社群与粉丝经济的结合

相关系列文章 电商技术揭秘相关系列文章合集&#xff08;1&#xff09; 电商技术揭秘相关系列文章合集&#xff08;2&#xff09; 电商技术揭秘相关系列文章合集&#xff08;3&#xff09; 电商技术揭秘四十一&#xff1a;电商平台的营销系统浅析 电商技术揭秘四十二&#…

Jboss 反序列化 CVE-2017-12149

一、漏洞简介 JBoss是一个管理EJB的容器和服务器&#xff0c;支持EJB 1.1、EJB 2.0和EJB3的规范。在/invoker/readonly路径下&#xff0c;攻击者可以构造序列化代码传入服务器进行反序列化,由于没有对反序列化操作进行任何检测&#xff0c;导致攻击者可以执行任意代码。 而jbo…

重发布和路由策略实验(课堂练习)

需求&#xff1a; 将1.1.1.0/24网段&#xff08;不在OSPF中&#xff09;重发布到网络中&#xff0c;不允许出现次优路径&#xff0c;实现全网可达。 需求分析&#xff1a; 1、在R1上重发布1.1.1.0/24网段&#xff0c;但是需要过滤192.168.12.0/24和192.168.13.0/24 2、在R2和R3…

考研数学|强化阶段怎么刷《660》《880》《1000》?

强化阶段想要刷好题&#xff0c;首先要选一本适合自己的题集&#xff01; 一般在强化阶段&#xff0c;大家用多个最多的题集就是660题&#xff0c;880题还有1000题 660题的特点是只训练客观题&#xff0c;虽然题目的质量很高&#xff0c;但是训练面还是比较窄 880题是综合训…

关于DDD和COLA的一些总结和思考

1|0思维&#xff1a;面向对象和面向过程 领域驱动设计本质上是讲的面向对象&#xff0c;但是谈面向对象&#xff0c;始终无法绕开面向过程&#xff0c;所以我们先好好说一下面向过程和面向对象这两个概念。 什么是面向过程呢&#xff0c;其实就是我们学习编程时最初被植入的逻辑…

C++笔记(STL标准库)

1.STL六大部件 容器 Containers分配器 Allocators&#xff1a;帮容器分配内存算法 Algorithms迭代器 Iterators&#xff1a;算法通过迭代器操作容器里的数据&#xff0c;是一种泛化的指针适配器 Adapters&#xff1a;修改或扩展已有类或函数的接口以满足特定的需求仿函数 Func…

《动手学深度学习》V2(11-18)

文章目录 十一、二 模型选择与过拟合和欠拟合1、模型的选择2、过拟合和欠拟合3、估计模型容量4、线性分类器的VC维5、过拟合欠拟合的代码实现 :fire:①生成数据集②定义评估损失③定义训练函数④三阶多项式函数拟合⑤线性函数拟合(欠拟合)⑤高阶多项式函数拟合(过拟合) 十三、权…

【020】基于JavaWeb实现的批报管理系统

项目介绍 基于jspservlet实现的批报管理系统采用B/S架构,该项目设计了一个角色管理员&#xff0c;管理员实现了我的案件、查询统计、项目维护等三大功能模块 技术栈 开发工具&#xff1a;Idea2020.3 运行环境&#xff1a;jdk1.8tomcat9.0mysql5.7 服务端技术&#xff1a;j…

初识C语言——第十八天

循环while/do while while 语法结构 while(表达式) 循环语句; break:在while循环中&#xff0c;break用于永久的终止循环 continue:在while循环中&#xff0c;continue的作用是跳过本次循环continue后面的代码 直接去判断部分&#xff0c;看是否进行下一次循环。 注意事项…