news 2026/6/23 7:00:15

(新卷,100分)- 最大矩阵和(Java JS Python C)

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
(新卷,100分)- 最大矩阵和(Java JS Python C)

(新卷,100分)- 最大矩阵和(Java & JS & Python & C)

题目描述

给定一个二维整数矩阵,要在这个矩阵中选出一个子矩阵,使得这个子矩阵内所有的数字和尽量大,我们把这个子矩阵称为和最大子矩阵,子矩阵的选取原则是原矩阵中一块相互连续的矩形区域。

输入描述

输入的第一行包含2个整数n, m(1 <= n, m <= 10),表示一个n行m列的矩阵,下面有n行,每行有m个整数,同一行中,每2个数字之间有1个空格,最后一个数字后面没有空格,所有的数字的在[-1000, 1000]之间。

输出描述

输出一行一个数字,表示选出的和最大子矩阵内所有的数字和。

用例
输入

3 4
-3 5 -1 5
2 4 -2 4
-1 3 -1 3

输出20
说明一个3*4的矩阵中,后面3列的子矩阵求和加起来等于20,和最大。
题目分析

看到这个题目标题,我很容易就联想到了最大子数组和

区别在于最大子数组和是一维的,而最大子矩阵和是二维的。

那么是不是有可能将最大子矩阵和的求解转成一维的呢?

下面是子矩阵可能存在的区域,即一行子矩阵,两行子矩阵,三行子矩阵

进一步简化,可得下图,即对求解下面每个区域的最大子矩阵

此时因为子矩阵的行数已经确定,因此我们可以将多行压缩为一行

此时对于最大子矩阵和的求解,就变为了最大子数组和的求解。

而最大子数组和的求解的状态转移方程我们已经在前一小结总结出来了:

dp[i] = max(dp[i-1], 0) + nums[i]。

还有一个难点就是二维数组压缩为一维数组的问题,解决思路如下,获取二维数组的行数rows、列数cols,创建一个长度为cols的一维数组,然后将二维数组双重for循环,外层遍历cols,内层遍历rows,这样每次循环就可以得到二维数组一个列上的所有元素,然后求和存入一维数组中,实现如下

function matrixZip(matrix) { let cols = matrix[0].length; let rows = matrix.length; let zip = new Array(cols).fill(0); for (let c = 0; c < cols; c++) { for (let r = 0; r < rows; r++) { zip[c] += matrix[r][c]; } } return zip; }
JavaScript算法源码
/* JavaScript Node ACM模式 控制台输入获取 */ const readline = require("readline"); const rl = readline.createInterface({ input: process.stdin, output: process.stdout, }); let lines = []; let n, m; rl.on("line", (line) => { lines.push(line); // 输入第一行时,提取出m、n if (lines.length === 1) { [n, m] = lines[0].split(" ").map((ele) => parseInt(ele)); } // 输入第一行后,再输入n行时,则开始启动算法程序 if (lines.length - 1 === n) { // 干掉第一行输入,即lines中存储的全是就是matrix要素 lines.shift(); // matrix是算法程序的入参二维数组 let matrix = []; // 将多行输入的matrix要素提取出来存到二维数组中 lines.forEach((line) => { matrix.push( line .split(" ") .map((ele) => parseInt(ele)) .slice(0, m) ); }); // 调用算法程序 console.log(maxSubMatrixSum(matrix)); // 将输入归0,重新接收下一轮 lines.length = 0; } }); function maxSubMatrixSum(matrix) { let dp = []; for (let i = 0; i < matrix.length; i++) { dp.push(maxSubArraySum(matrix[i])); for (let j = i + 1; j < matrix.length; j++) { dp.push(maxSubArraySum(matrixZip(matrix.slice(i, j + 1)))); } } return dp.sort((a, b) => b - a)[0]; } function maxSubArraySum(nums) { let dp = new Array(nums.length); let result = (dp[0] = nums[0]); for (let i = 1; i < nums.length; i++) { dp[i] = Math.max(dp[i - 1], 0) + nums[i]; result = Math.max(result, dp[i]); } return result; } function matrixZip(matrix) { let cols = matrix[0].length; let rows = matrix.length; let zip = new Array(cols).fill(0); for (let c = 0; c < cols; c++) { for (let r = 0; r < rows; r++) { zip[c] += matrix[r][c]; } } return zip; }
Java算法源码
import java.util.ArrayList; import java.util.Arrays; import java.util.Scanner; public class Main { public static void main(String[] args) { Scanner sc = new Scanner(System.in); int n = sc.nextInt(); int m = sc.nextInt(); int[][] matrix = new int[n][m]; for (int i = 0; i < n; i++) { for (int j = 0; j < m; j++) { matrix[i][j] = sc.nextInt(); } } System.out.println(getResult(n, m, matrix)); } public static int getResult(int n, int m, int[][] matrix) { ArrayList<Integer> dp = new ArrayList<>(); for (int i = 0; i < n; i++) { dp.add(maxSubArraySum(matrix[i])); // 一行子矩阵最大和 for (int j = i + 1; j < n; j++) { dp.add(maxSubArraySum(matrixZip(Arrays.copyOfRange(matrix, i, j + 1)))); // 多行子矩阵最大和 } } return dp.stream().max((a, b) -> a - b).orElse(0); // 求出最大和 } // 最大子数组和求解 public static int maxSubArraySum(int[] nums) { int[] dp = new int[nums.length]; int res = dp[0] = nums[0]; for (int i = 1; i < nums.length; i++) { dp[i] = Math.max(dp[i - 1], 0) + nums[i]; res = Math.max(res, dp[i]); } return res; } // 多行子矩阵,压缩为一行子数组 public static int[] matrixZip(int[][] matrix) { int cols = matrix[0].length; int rows = matrix.length; int[] zip = new int[cols]; for (int c = 0; c < cols; c++) { for (int r = 0; r < rows; r++) { zip[c] += matrix[r][c]; } } return zip; } }
Python算法源码
# 输入获取 n, m = map(int, input().split()) matrix = [list(map(int, input().split())) for i in range(n)] # 最大子数组和求解 def maxSubArraySum(nums): dp = [0 for i in range(len(nums))] res = dp[0] = nums[0] for i in range(1, len(nums)): dp[i] = max(dp[i - 1], 0) + nums[i] res = max(res, dp[i]) return res # 将多行子矩阵,压缩为一维数组 def matrixZip(matrix): cols = len(matrix[0]) rows = len(matrix) zip = [0 for i in range(cols)] for c in range(cols): for r in range(rows): zip[c] += matrix[r][c] return zip # 算法入口 def getResult(n, m, matrix): dp = [] for i in range(n): dp.append(maxSubArraySum(matrix[i])) for j in range(i + 1, n): dp.append(maxSubArraySum(matrixZip(matrix[i:j + 1]))) dp.sort() return dp[-1] # 算法调用 print(getResult(n, m, matrix))
C算法源码
#include <stdio.h> #include <stdlib.h> #include <limits.h> #define MAX(a,b) ((a) > (b) ? (a) : (b)) int getResult(int n, int m, int matrix[n][m]); int maxSubArraySum(int nums[], int nums_size); int* matrixZip(int rowFrom, int rowTo, int rows, int cols, int matrix[rows][cols]); int main() { int n, m; scanf("%d %d", &n, &m); int matrix[n][m]; for (int i = 0; i < n; i++) { for (int j = 0; j < m; j++) { scanf("%d", &matrix[i][j]); } } printf("%d\n", getResult(n, m, matrix)); return 0; } int getResult(int n, int m, int matrix[n][m]) { int ans = INT_MIN; for(int i=0; i<n; i++) { ans = MAX(ans, maxSubArraySum(matrix[i], m)); // 一行子矩阵最大和 for(int j=i+1; j<n; j++) { ans = MAX(ans, maxSubArraySum(matrixZip(i, j, n, m, matrix), m)); // 多行子矩阵最大和 } } return ans; } // 最大子数组和求解 int maxSubArraySum(int nums[], int nums_size) { int dp[nums_size]; int res = dp[0] = nums[0]; for(int i=1; i<nums_size; i++) { dp[i] = MAX(dp[i-1], 0) + nums[i]; res = MAX(res, dp[i]); } return res; } // 多行子矩阵,压缩为一行子数组 int* matrixZip(int rowFrom, int rowTo, int rows, int cols, int matrix[rows][cols]) { int* zip = (int*) calloc(cols, sizeof(int)); for(int c = 0; c < cols; c++) { for(int r = rowFrom; r <= rowTo; r++) { zip[c] += matrix[r][c]; } } return zip; }
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/23 4:49:20

实习面试题-JavaScript 面试题

1.JavaScript 有哪些数据类型?它们的区别是什么? JavaScript 有八种基本数据类型,分为原始类型(Primitive Types)和引用类型(Reference Types): 原始类型 1)Undefined:表示变量未初始化。一个变量声明后但未赋值时,它的默认值是 undefined。 2)Null:表示一个空…

作者头像 李华
网站建设 2026/6/23 19:46:58

解决‘此扩展程序不再受支持’问题:FLUX.1-dev开发环境兼容性优化方案

FLUX.1-dev开发环境兼容性优化&#xff1a;从问题到实践的深度解析 在浏览器插件开发的世界里&#xff0c;一个看似无害的提示——“此扩展程序不再受支持”——往往能让整个项目陷入停滞。尤其是当它出现在你基于最新AI模型构建的文生图工具中时&#xff0c;那种挫败感尤为强烈…

作者头像 李华
网站建设 2026/6/23 19:46:19

火山引擎AI大模型生态中FLUX.1-dev的独特定位分析

火山引擎AI大模型生态中FLUX.1-dev的独特定位分析 在AIGC浪潮席卷内容创作领域的今天&#xff0c;一个核心问题始终困扰着从业者&#xff1a;如何让AI真正“听懂”复杂的视觉指令&#xff1f;无论是广告设计师反复修改提示词却得不到理想构图&#xff0c;还是电商平台需要批量生…

作者头像 李华
网站建设 2026/6/22 23:26:36

抖音直播回放永久保存指南:告别内容丢失的烦恼

抖音直播回放永久保存指南&#xff1a;告别内容丢失的烦恼 【免费下载链接】douyin-downloader 项目地址: https://gitcode.com/GitHub_Trending/do/douyin-downloader 还在为错过精彩直播而懊恼吗&#xff1f;&#x1f914; 当你看到心仪主播的直播&#xff0c;想要永…

作者头像 李华
网站建设 2026/6/23 19:46:59

Bypass Paywalls Clean完整使用教程:快速解锁全网付费内容

Bypass Paywalls Clean是一款专为Chrome浏览器设计的强大扩展工具&#xff0c;能够智能绕过各类网站的付费墙限制&#xff0c;让您免费访问原本需要付费订阅的优质内容。无论您是新闻阅读者、学术研究者还是商业分析师&#xff0c;这款工具都能为您提供便捷的内容获取体验。 【…

作者头像 李华
网站建设 2026/6/23 5:08:47

国产CAD实现铸造与热处理工艺的标准化控制

铸造、热处理等特种工艺&#xff0c;其质量在很大程度上依赖于对过程参数&#xff08;如温度、时间&#xff09;的精确控制。过去&#xff0c;这些参数多依赖于老师傅的个人经验&#xff0c;存在波动性。为实现质量的稳定与均一&#xff0c;必须将个人经验转化为可重复、可验证…

作者头像 李华