码农知识堂 - 1000bd
  •   Python
  •   PHP
  •   JS/TS
  •   JAVA
  •   C/C++
  •   C#
  •   GO
  •   Kotlin
  •   Swift
  • AcWing 4405. 统计子矩阵(每日一题)


    如果你觉得这篇题解对你有用,可以点点关注再走呗~

    题目描述

    给定一个 N×M 的矩阵 A,请你统计有多少个子矩阵 (最小 1×1,最大 N×M) 满足子矩阵中所有数的和不超过给定的整数 K ?
    输入格式

    第一行包含三个整数 N,M 和 K。

    之后 N 行每行包含 M 个整数,代表矩阵 A。
    输出格式

    一个整数代表答案。

    数据范围

    对于 30%
    的数据,N,M≤20,
    对于 70% 的数据,N,M≤100,
    对于 100% 的数据,1≤N,M≤500;0≤Aij≤1000;1≤K≤2.5×108。
    输入样例:

    3 4 10
    1 2 3 4
    5 6 7 8
    9 10 11 12

    输出样例:

    19

    样例解释

    满足条件的子矩阵一共有 19,包含:大小为 1×1的有 10个。
    大小为 1×2的有 3个。
    大小为 1×3的有 2个。
    大小为 1×4的有 1个。
    大小为 2×1的有 3 个。

    分析

    直接暴力枚举
    需要枚举上、下、左、右 四个边界
    时间复杂度 O(n^4) = O(10^10)
    n=500 500^4=62500000000=6.25x10^10必
    定TLE
    我们看能不能优化成三维去做?
    500^3=125000000=1.25*10^8
    这样看还是过不了
    于是引入双指针算法将枚举次数降为1.25*10^8/2
    时间复杂度为**O(6*10^7)**
    这样便可以过了

    优化

    双指针算法+前缀和
    题目问满足子矩阵中所有数的和不超过给定的整数 K的子矩阵
    所以我们用前缀和去处理矩阵内所有数的和
    时间复杂度为**O(1)**

    双指针怎么实现?

    下面让我带你来分析
    我们需要用到4个变量:
    up(矩阵的上界)
    down(矩阵的下界)
    l(矩阵的左界)
    r(矩阵的右界)

    在这里插入图片描述

    先固定上下界,现在上下界固定下来了。

    在这里插入图片描述

    再依次去枚举上下界,每次枚举上下界时移动左右边界。

    在这里插入图片描述

    我们需要统计的是左右边界移动的矩阵内有多少列

    有多少列就有多少个满足条件的矩阵

    矩阵个数:r-l+1
    为什么统计的是**矩阵内的列数****见盲点分析

    怎么样去移动左右边界?
    在这里插入图片描述

    进一步--------------
    在这里插入图片描述

    先固定右边界,再去枚举左边界

    过程分析

    左边界l满足当左边界 l 到右边界 r 这一矩阵内所有数的和> K 时。
    说明我们的 l~r 这块矩阵不能包含太多的数,l 需要往右移动。
    l 一直移动下去,直至 l 移动到 l~r 这块矩阵的所有数的和值<= k 时,l停止。

    在计算完这块矩阵内列的矩阵个数后,再去移动r,依次类推。
    确保每次 l~r 这块矩阵内的所有数的和均<=k
    这样便可以将矩阵的个数不重不漏的计算出来。
    在这里插入图片描述

    注意

    怕很多同学不清楚每一列(个)矩阵代表一列(个)数字
    l、r也是一列矩阵,只不过为了便于理解,将l、r抽象成两条线/边界来看
    其实l、r本身是一列,是有数字的!!!
    下面的分析与此表述同步!
    数字图如下:
    在这里插入图片描述

    盲点分析

    为什么去枚举l~r的列数?
    我们在枚举时枚举的并不全是向前面的图形那样
    上面的图形描绘的是整体的矩阵效果,便于理解双指针的移动。

    下面我们来看看枚举时的过程和细节
    在这里插入图片描述

    枚举的过程可以抽象看成是在一整个矩阵内拖动橙色长方形,从右上往左下拉。
    实际上是由很多的l、r不断分割矩阵的数,见下图:
    在这里插入图片描述

    注:图形是便于理解,每一条线不代表一种情况,实际计算时不会重复累加。

    从图中便可以看到,我们将图形理解成动态的过程,从左上到右下不断划分出小矩阵,再去判断是否满足条件,从而将矩阵的总数加起来。
    像图中的蓝色小矩阵即是枚举是每一列的矩阵个数,依次移动指针,累加矩阵个数。

    注意:l、r也是一列矩阵,是占据了数字的一列。

    为什么计算的是列数?

    在这里插入图片描述

    枚举时计算的是每一列的矩阵数,这是因为我们是从右上往左下移动。
    如果统计的是每一行的矩阵数则同一行的矩阵数都被并入最终的结果其中,造成结果的错漏,而从每一行去统计矩阵数也与我们枚举时移动的方向不一致,因此不能是统计每一行的矩阵数。同学们仔细想一下对不对?

    关于r-l+1的由来:

    先看最普遍的情况:

    在这里插入图片描述

    由上图:l在左边,r在右边,中间间隔3列,即3个矩阵,再加上两列其本身所在的那一列矩阵。总共是3+2=5个矩阵,代入公式检验一下,检查一下对不对?
    根据图形,假设r=5,l=1,中间的列数分别为2、3、4总共3列
    r-l+1=5-1+1=5 结果正确!

    再看边界情况:

    当r=l时,代入公式r-l+1=0+1=1。
    得出1个,而这个便是r、l所在这一列的矩阵,见下图:
    在这里插入图片描述

    注:每一个矩阵代表一个数字

    Accode

    import java.util.*;
    public class Main{
    	static int N=510;
    	static int [][]a=new int[N][N];
    	static int [][]s=new int[N][N];
    	static int n,m,k;
    	public static void main(String []args) {
    	Scanner sc=new Scanner(System.in);
    	n=sc.nextInt();
    	m=sc.nextInt();
    	k=sc.nextInt();
    	//预处理前缀和,便于计算矩阵内所有数的和值
    	for(int i=1;i<=n;i++) {
    		for(int j=1;j<=m;j++) {
    			a[i][j]=sc.nextInt();
    			s[i][j]=s[i-1][j]+s[i][j-1]-s[i-1][j-1]+a[i][j];
    					}
    	}
    	
    	long res=0;
    	//枚举上下边界
    	for(int up=1;up<=n;up++) {
    		for(int down=up;down<=n;down++) {
    		    
    			int l=1;//左边界
    			int r=1;//右边界
    			
    			while(r<=m) {
    			    //右边界在整个矩阵内
    			    
    				while(l<=r&&!check(up,l,down,r))l++;
    				//不满足和值小于等于k则移动左边界
    				
    				res+=(long)r-l+1;
    				//统计列数
    				r++;
    				//左边界固定下来,可以移动右边界
    			}
    			
    		}
    	}
    	System.out.println(res);	
    	}
    	static boolean check(int x1,int y1,int x2,int y2) {
    		//前缀和
    		long sum=s[x2][y2]-s[x1-1][y2]-s[x2][y1-1]+s[x1-1][y1-1];
    		//判断是否满足sum小于等于k这一条件
    		return sum<=k;
    	}
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50

    往期回顾

    不清楚蓝桥杯考什么的点点下方👇

    考点秘籍

    想背纯享模版的伙伴们点点下方👇

    蓝桥杯省一你一定不能错过的模板大全(第一期)

    蓝桥杯省一你一定不能错过的模板大全(第二期)

    蓝桥杯省一你一定不能错过的模板大全(第三期)

    蓝桥杯省一你一定不能错过的模板大全(第四期)!!!

    想背注释模版的伙伴们点点下方👇

    蓝桥杯必背第一期

    蓝桥杯必背第二期

    往期精彩回顾

    蓝桥杯上岸每日N题 第一期(一)!!!

    蓝桥杯上岸每日N题第一期(二)!!!

    蓝桥杯上岸每日N题第一期(三)!!!

    蓝桥杯上岸每日N题第二期(一)!!!

    蓝桥杯上岸每日N题第三期(一)!!!

    蓝桥杯上岸每日N题 第四期(最少刷题数)!!!

    蓝桥杯上岸每日N题 第五期(山)!!!

    蓝桥杯上岸每日N题 第六期(求阶乘)!!!

    蓝桥杯上岸每日N题 第七期(小猫爬山)!!!

    蓝桥杯上岸每日N题 第八期 (全球变暖)!!!

    蓝桥杯每日N题 (消灭老鼠)

    蓝桥杯每日N题(杨辉三角形)

    蓝桥杯每日N题 (砝码称重)

    蓝桥杯上岸每日N题(鸡尾酒)

    操作系统期末题库 第九期(完结)

    LeetCode Hot100 刷题(第三期)

    idea创建SpringBoot项目报错解决方案

    数据库SQL语句(期末冲刺)

    想看JavaB组填空题的伙伴们点点下方 👇

    填空题

    竞赛干货

    算法竞赛字符串常用操作大全

    蓝桥杯上岸必刷!!!(模拟/枚举专题)

    蓝桥杯上岸必背!!! (第三期 DP)

    蓝桥杯上岸必背!!!(第四期DFS)

    蓝桥杯上岸必背!!!(第五期BFS)

    蓝桥杯上岸必背!!!(第六期树与图的遍历)

    蓝桥杯上岸必背!!!(第七期 最短路算法)

    蓝桥杯上岸必背!!!(第八期 简单数论)

    蓝桥杯上岸必刷!!!(进制、数位专题)

    蓝桥杯上岸考点清单 (冲刺版)!!!

    蓝桥杯上岸必背模板 (纯享版)

  • 相关阅读:
    通过内网穿透实现远程连接群晖Drive,轻松实现异地访问群晖NAS
    自己写了一个简易的android transformation.map
    Pushmall推熵平台--对电商未来5年发展趋势分析
    如何保证接口的幂等性?
    Linux下 Apache 配置 SSL 支持 https
    关于safari浏览器浏览html video标签无法正常播放的问题
    开关电源-交流220V降压电路-电阻电容降压原理
    【win11】 win11环境问题备忘
    数据通信——应用层(超文本)
    实验2.1.2 交换机的常用配置
  • 原文地址:https://blog.csdn.net/joeyoj/article/details/132591863
  • 最新文章
  • 攻防演习之三天拿下官网站群
    数据安全治理学习——前期安全规划和安全管理体系建设
    企业安全 | 企业内一次钓鱼演练准备过程
    内网渗透测试 | Kerberos协议及其部分攻击手法
    0day的产生 | 不懂代码的"代码审计"
    安装scrcpy-client模块av模块异常,环境问题解决方案
    leetcode hot100【LeetCode 279. 完全平方数】java实现
    OpenWrt下安装Mosquitto
    AnatoMask论文汇总
    【AI日记】24.11.01 LangChain、openai api和github copilot
  • 热门文章
  • 十款代码表白小特效 一个比一个浪漫 赶紧收藏起来吧!!!
    奉劝各位学弟学妹们,该打造你的技术影响力了!
    五年了,我在 CSDN 的两个一百万。
    Java俄罗斯方块,老程序员花了一个周末,连接中学年代!
    面试官都震惊,你这网络基础可以啊!
    你真的会用百度吗?我不信 — 那些不为人知的搜索引擎语法
    心情不好的时候,用 Python 画棵樱花树送给自己吧
    通宵一晚做出来的一款类似CS的第一人称射击游戏Demo!原来做游戏也不是很难,连憨憨学妹都学会了!
    13 万字 C 语言从入门到精通保姆级教程2021 年版
    10行代码集2000张美女图,Python爬虫120例,再上征途
Copyright © 2022 侵权请联系2656653265@qq.com    京ICP备2022015340号-1
正则表达式工具 cron表达式工具 密码生成工具

京公网安备 11010502049817号