蓝桥杯——统计子矩阵


解法:二维前缀和+双指针

代码:

c 复制代码
#include <iostream>
using namespace std;
typedef long long ll;
ll prefix[505][505], a[250010];
int main()
{
	ll n, m, k, ans = 0; cin >> n >> m >> k;
  for(int i = 1; i <= n; i++)
     for(int j = 1; j <= m; j++){
       cin >> prefix[i][j];
       prefix[i][j] += prefix[i-1][j] + prefix[i][j-1] - prefix[i-1][j-1];
       //cout << prefix[i][j] << endl;
     }
  //for(int i = 1; i < n*m; i++) cout << prefix[i] << endl; 
  for(int i = 1; i <= n; i++)
    for(int j = 1; j <= i; j++){
      for(int l = 1, r = 1; r <= m; r++){
        while(l <= r && prefix[i][r] - prefix[i][l-1] - prefix[j-1][r] + prefix[j-1][l-1] > k) l++;
        if(l <= r) ans += r-l+1; //每一次循环,以r为区间,r每次必++
      }
    }
      
  cout << ans;
  return 0;
}

代码解释:

这段代码的核心是计算满足条件的子矩阵的数量。让我们逐步分析代码的逻辑,特别是你提到的 if (l <= r) ans += r - l + 1; 这一行。

代码逻辑概述

  1. 二维前缀和数组

    • prefix[i][j] 表示从矩阵的左上角 (1,1) 到当前位置 (i,j) 的子矩阵的元素和。
    • 通过二维前缀和,可以在常数时间内计算任意子矩阵的和。
  2. 目标

    • 找出所有满足条件的子矩阵数量,其中子矩阵的和不超过给定的阈值 k

关键代码分析

外层循环
cpp 复制代码
for (int i = 1; i <= n; i++) // 枚举子矩阵的上边界
    for (int j = i; j <= n; j++) // 枚举子矩阵的下边界
  • 这两层循环枚举了所有可能的子矩阵的上下边界。i 是子矩阵的上边界,j 是子矩阵的下边界。
内层循环
cpp 复制代码
for (int l = 1, r = 1; r <= m; r++) // 枚举子矩阵的右边界
  • 这一层循环枚举了子矩阵的右边界 r,同时用 l 表示子矩阵的左边界。
  • lr 都是从 1 开始,r 逐渐向右扩展。
关键条件
cpp 复制代码
while (l <= r && prefix[j][r] - prefix[j][l-1] - prefix[i-1][r] + prefix[i-1][l-1] > k) l++;
  • 这一行的作用是通过滑动窗口的方式,找到满足条件的最小左边界 l
  • prefix[j][r] - prefix[j][l-1] - prefix[i-1][r] + prefix[i-1][l-1] 计算的是子矩阵 (i, l)(j, r) 的和。
  • 如果当前子矩阵的和大于 k,则需要缩小窗口,即将左边界 l 向右移动,直到子矩阵的和不超过 k
关键更新
cpp 复制代码
if (l <= r) ans += r - l + 1;
  • 这一行是关键所在。
  • 在前面的 while 循环中,已经通过调整左边界 l,使得子矩阵 (i, l)(j, r) 的和不超过 k
  • 如果 l <= r,说明当前窗口是有效的,即存在满足条件的子矩阵。
  • r - l + 1 表示在当前的上下边界 (i, j) 和右边界 r 的情况下,所有可能的左边界 l 的数量。
举例说明

假设当前上下边界为 (i, j),右边界为 r,左边界 l 从 1 开始:

  • 如果子矩阵 (i, 1)(j, r) 的和不超过 k,那么子矩阵 (i, 2)(j, r)(i, 3)(j, r) 等等也一定满足条件。
  • 因此,所有满足条件的子矩阵数量是从 lr 的所有可能的左边界数量,即 r - l + 1

总结

if (l <= r) ans += r - l + 1; 这一行的作用是:

  • 在当前上下边界 (i, j) 和右边界 r 的情况下,统计所有满足条件的子矩阵数量。
  • 这些子矩阵的左边界从 lr,数量为 r - l + 1
相关推荐
Zfox_1 小时前
CANN Catlass 算子模板库深度解析:高性能矩阵乘(GEMM)原理、融合优化与模板化开发实践
线性代数·矩阵
觉醒大王2 小时前
AI写的青基中了
人工智能·笔记·深度学习·学习·职场和发展·学习方法
程序员敲代码吗2 小时前
面试中sessionStorage问题引发深度探讨
面试·职场和发展
橘颂TA3 小时前
【测试】高效浏览器操作:基础功能与优化设置大全
c++·功能测试·职场和发展·测试·web测试
历程里程碑3 小时前
普通数组----合并区间
java·数据结构·python·算法·leetcode·职场和发展·tornado
iAkuya3 小时前
(leetcode)力扣100 61分割回文串(回溯,动归)
算法·leetcode·职场和发展
VT.馒头4 小时前
【力扣】2695. 包装数组
前端·javascript·算法·leetcode·职场和发展·typescript
源代码•宸4 小时前
大厂技术岗面试之谈薪资
经验分享·后端·面试·职场和发展·golang·大厂·职级水平的薪资
马猴烧酒.5 小时前
【面试八股|JVM虚拟机】JVM虚拟机常考面试题详解
jvm·面试·职场和发展
iAkuya6 小时前
(leetcode)力扣100 62N皇后问题 (普通回溯(使用set存储),位运算回溯)
算法·leetcode·职场和发展