LeetCode 每日一题笔记 日期:2025.03.25 题目:3546.等和矩阵分割

LeetCode 每日一题笔记

0. 前言

  • 日期:2025.03.25
  • 题目:3546.等和矩阵分割
  • 难度:中等
  • 标签:数组 矩阵 前缀和

1. 题目理解

问题描述

给你一个由正整数组成的 m x n 矩阵 grid。你的任务是判断是否可以通过 一条水平或一条垂直分割线 将矩阵分割成两部分,使得:

分割后形成的每个部分都是 非空 的。

两个部分中所有元素的和 相等。

如果存在这样的分割,返回 true;否则,返回 false。

示例

输入: grid = \[1,4,2,3]

输出: true

解释:矩阵总和为 10,目标和为 5。

水平分割第一行,和为 5,满足条件。

2. 解题思路

核心观察

  • 要将矩阵分成和相等的两部分,总和必须是偶数,否则直接返回 false;
  • 水平分割:从上到下累加行和,若某一行累加和等于总和的一半,则可分割;
  • 垂直分割:从左到右累加列和,若某一列累加和等于总和的一半,则可分割;
  • 矩阵元素全为正整数,累加和单调递增,无需考虑重复或回退。

算法步骤

  1. 计算矩阵所有元素的总和 total;
  2. 若 total 是奇数,直接返回 false;
  3. 计算目标和 target = total / 2;
  4. 从上到下逐行累加,判断是否存在行前缀和等于 target;
  5. 从左到右逐列累加,判断是否存在列前缀和等于 target;
  6. 满足任一条件返回 true,否则返回 false。

3. 代码实现

java 复制代码
class Solution {
    public boolean canPartitionGrid(int[][] grid) {
        int n = grid.length;
        int m = grid[0].length;
        long total = 0;
        for (int i = 0; i < n; i++) {
            for (int j = 0; j < m; j++) {
                total += grid[i][j];
            }
        }
        if (total % 2 != 0) return false;
        long target = total / 2;
        long sum = 0;
        for (int i = 0; i < n; i++) {
            for (int j = 0; j < m; j++) {
                sum += grid[i][j];
            }
            if (sum == target) {
                return true;
            }
        }
          long colSum = 0;
        for (int j = 0; j < m; j++) {
            for (int i = 0; i < n; i++) {
                colSum += grid[i][j];
            }
            if (colSum == target) return true;
        }
        return false;
    }
}

4. 代码优化说明

优化点:提前终止遍历

由于元素均为正整数,当前累加和超过 target 时可直接终止当前方向的遍历,减少无效计算:

java 复制代码
class Solution {
    public boolean canPartitionGrid(int[][] grid) {
        int n = grid.length;
        int m = grid[0].length;
        long total = 0;
        for (int[] row : grid) {
            for (int num : row) total += num;
        }
        if (total % 2 != 0) return false;
        long target = total / 2;

        long rowSum = 0;
        for (int[] row : grid) {
            for (int num : row) rowSum += num;
            if (rowSum == target) return true;
            if (rowSum > target) break;
        }

        long colSum = 0;
        for (int j = 0; j < m; j++) {
            for (int i = 0; i < n; i++) colSum += grid[i][j];
            if (colSum == target) return true;
            if (colSum > target) break;
        }
        return false;
    }
}

5. 复杂度分析

  • 时间复杂度:(O(m \times n))

    • 遍历矩阵计算总和:(O(mn));
    • 水平/垂直遍历:各一次完整矩阵遍历,总复杂度线性。
  • 空间复杂度:(O(1))

    • 仅使用常量级额外空间。

6. 总结

  • 核心思路:总和判断 + 行/列前缀和验证
  • 关键技巧:利用正整数单调性,只需一次顺序遍历即可判断可分割性;
  • 适用场景:单条水平/垂直线分割、两部分非空、和相等的矩阵判定问题。

关键点回顾

  1. 总和必须为偶数是可分割的必要条件;
  2. 只需检查行前缀和、列前缀和是否等于总和一半;
  3. 正整数保证累加和单调,可提前终止无效遍历。
相关推荐
To_OC3 小时前
LC 1 两数之和:面试第一道必考题,暴力解法直接被面试官 pass
javascript·算法·leetcode
LinXunFeng2 天前
Obsidian - 使用 Share Note 分享笔记并自部署
前端·笔记·github
Bobolink_6 天前
TikTok矩阵账号如何批量养号?工作室级运营方案分享
矩阵·内容运营·跨境电商·tik tok·账号运营
闪闪发亮的小星星6 天前
高斯光以及高斯光公式解释
笔记
cqbzcsq6 天前
CellFlow虚拟细胞论文阅读
论文阅读·人工智能·笔记·学习·生物信息
阿米亚波6 天前
【Windows】QEMU 启动 openEuler aarch64/arm64 架构系统 + 离线软件源
linux·windows·经验分享·笔记·架构·arm
自传.6 天前
尚硅谷 Vibe Coding|第三章(1) Claude Code深度使用与进阶技巧 学习笔记
笔记·学习·尚硅谷·vibecoding
想吃火锅10056 天前
【leetcode】121.买卖股票的最佳时机js/c++
算法·leetcode·职场和发展
.千余6 天前
【C++】模板进阶全解:非类型参数|全特化|偏特化|分离编译完全指南
开发语言·c++·笔记·学习·其他
自传.6 天前
尚硅谷 Vibe Coding|第二章 AI编程工具生态 学习笔记
笔记·学习·ai编程·尚硅谷·vibe coding