蓝桥杯——统计子矩阵


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

代码:

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
相关推荐
Bobolink_16 天前
TikTok矩阵账号如何批量养号?工作室级运营方案分享
矩阵·内容运营·跨境电商·tik tok·账号运营
dayuOK630716 天前
写作卡壳怎么办?我的“5分钟启动法”
人工智能·职场和发展·自动化·新媒体运营·媒体
枫子有风16 天前
LLM-Agent智能体(大厂面试常问)
面试·职场和发展·llm·agent
重生之后端学习16 天前
Java入门
java·开发语言·职场和发展
AIHR数智引擎16 天前
KPI物理失效:AI原生组织的效能重构与技能度量
人工智能·经验分享·职场和发展·重构·ai-native·aihr
想吃火锅100516 天前
【leetcode】121.买卖股票的最佳时机js/c++
算法·leetcode·职场和发展
程序员小远16 天前
自动化测试基础知识总结
自动化测试·软件测试·python·selenium·测试工具·职场和发展·测试用例
嘿黑嘿呦16 天前
chap 8排序
算法·蓝桥杯·排序算法·软件工程
AI_yangxi16 天前
短视频矩阵系统专业公司
大数据·人工智能·矩阵
小欣加油17 天前
leetcode3612 用特殊操作处理字符串I
数据结构·c++·算法·leetcode·职场和发展