丹青千秋酿,一醉解愁肠。

无悔少年枉,只愿壮志狂

题目

题目描述

小 F 很喜欢数学,但是到了高中以后数学总是考不好。

有一天,他在数学课上发起了呆;他想起了过去的一年。一年前,当他初识算法竞赛的 时候,觉得整个世界都焕然一新。这世界上怎么会有这么多奇妙的东西?曾经自己觉得难以 解决的问题,被一个又一个算法轻松解决。

小 F 当时暗自觉得,与自己的幼稚相比起来,还有好多要学习的呢。

一年过去了,想想都还有点恍惚。

他至今还能记得,某天晚上听着入阵曲,激动地睡不着觉,写题写到鸡鸣时分都兴奋不 已。也许,这就是热血吧。

也就是在那个时候,小 F 学会了矩阵乘法。让两个矩阵乘几次就能算出斐波那契数列的 第 $10^{100}$ 项,真是奇妙无比呢。

不过,小 F 现在可不想手算矩阵乘法——他觉得好麻烦。取而代之的,是一个简单的小 问题。他写写画画,画出了一个 $n×m$的矩阵,每个格子里都有一个不超过 k 的正整数。

小 F 想问问你,这个矩阵里有多少个不同的子矩形中的数字之和是 k 的倍数? 如果把一个子矩形用它的左上角和右下角描述为 $(x_1,y_1,x_2,y_2)$,其中$x_1 \le x_2,y_1 \le y_2$ ​:那么,我们认为两个子矩形是不同的,当且仅当他们以 $(x_1,y_1,x_2,y_2)$ 表示时不同;也就是 说,只要两个矩形以 $(x_1,y_1,x_2,y_2)$表示时相同,就认为这两个矩形是同一个矩形,你应该 在你的答案里只算一次。

输入输出格式

输入格式:

从标准输入中读入数据。

输入第一行,包含三个正整数 n,m,k;

输入接下来 n 行,每行包含 m 个正整数,第 i 行第 j 列表示矩阵中第 i 行第 j 列 中所填的正整数 a_{i,j}。

输出格式:

输出到标准输出中。

输入一行一个非负整数,表示你的答案。

样例

Sample Input

  1. 2 3 2
  2. 1 2 1
  3. 2 1 2

Sample Output

  1. 6

数据范围与说明

样例说明
这些矩形是符合要求的:
(1, 1, 1, 3),(1, 1, 2, 2),(1, 2, 1, 2),(1, 2, 2, 3),(2, 1, 2, 1),(2, 3, 2, 3)

数据范围

题解

分析

这道题上来先来一个 $O(N^4)$ 的暴力,使用前缀和;

但是,我们会发现,在枚举每个子矩阵时,有的部分是重复计算的

  • i,j枚举子矩阵的上下边界,o枚举我们处理到了第几列,把i、j行之间,右边界是第o列的矩阵压成一个数,之后枚举统计。

  • 我们要把 i 到 j 的矩阵变成一行,然后就可以做 (K倍区间) 详细的可以看

链接 http://blog.csdn.net/qq_35776409/article/details/78226120

具体原理:

对于任意一段区间[l,r]的和就是sum[r]-sum[l-1].

(sum[r]-sum[l-1])%k 保证了[l,r]这段区间要么%k等于0 要么比k小.

等于0这表示了正好是k的倍数 然后通过前缀和相同的数据来判断出剩下的k的倍数:$(sum[r]-sum[l-1])%k == 0$.

变形后就是:sum[r]%k==sum[l-1]%k

代码

  1. #include<iostream>
  2. #include<cstring>
  3. #include<cstdio>
  4.  
  5. using namespace std;
  6. #define MAXN 410
  7. typedef long long LL;
  8. int a[MAXN][MAXN],b[MAXN],cnt[1000005];
  9. LL sum[MAXN][MAXN];
  10. inline void read(int &x){
  11. x=0; int f=1; char c=getchar();
  12. while(c>'9'||c<'0'){ if(c=='-')f=-1; c=getchar(); }
  13. while(c>='0'&&c<='9'){ x=x*10+c-'0'; c=getchar(); } x*=f;
  14. }
  15.  
  16. LL Ans = 0;
  17. int main(){
  18. freopen("rally.in","r",stdin);
  19. freopen("rally.out","w",stdout);
  20. int n,m,Mod;
  21. read(n),read(m),read(Mod);
  22. for(int i=1; i<=n; ++i)
  23. for(int j=1; j<=m; ++j){
  24. read(a[i][j]);
  25. sum[i][j] = (sum[i-1][j] + a[i][j] + sum[i][j-1] - sum[i-1][j-1]);
  26. if(sum[i][j] >= Mod) sum[i][j] -= Mod;
  27. }
  28. for(int i=0; i<n; ++i)
  29. for(int j=i+1; j<=n; ++j){
  30. cnt[0] = 1;
  31. for(int k=1; k<=m; ++k) {
  32. b[k] = (sum[j][k] - sum[i][k] ) % Mod;
  33. if(b[k] < 0) b[k] += Mod;
  34. Ans += cnt[b[k]];
  35. ++cnt[b[k]];
  36. }
  37. for(int k=1; k<=m; ++k) cnt[b[k]] = 0;
  38. }
  39. printf("%lld\n",Ans);
  40. fclose(stdin); fclose(stdout);
  41. return 0;
  42. }

  

【题解】入阵曲 luogu3941 前缀和 压维的更多相关文章

  1. BZOJ 2101 [Usaco2010 Dec]Treasure Chest 藏宝箱:区间dp 博弈【两种表示方法】【压维】

    题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=2101 题意: 共有n枚金币,第i枚金币的价值是w[i]. 把金币排成一条直线,Bessie ...

  2. [luogu]P3941 入阵曲[前缀和][压行]

    [luogu]P3941 入阵曲 题目描述 小 F 很喜欢数学,但是到了高中以后数学总是考不好. 有一天,他在数学课上发起了呆:他想起了过去的一年.一年前,当他初识算法竞赛的 时候,觉得整个世界都焕然 ...

  3. [洛谷P3941]:入阵曲(前缀和+桶)

    题目传送门 题目背景 丹青千秋酿,一醉解愁肠.无悔少年枉,只愿壮志狂. 题目描述 小$F$很喜欢数学,但是到了高中以后数学总是考不好.有一天,他在数学课上发起了呆:他想起了过去的一年.一年前,当他初识 ...

  4. Luogu P3941 入阵曲【前缀和】By cellur925

    题目传送门 题目大意:给你一个\(n\)*\(m\)的矩阵,每个位置都有一个数,求有多少不同的子矩阵使得矩阵内所有数的和是\(k\)的倍数. 数据范围给的非常友好233,期望得到的暴力分:75分.前1 ...

  5. BZOJ1177 [Apio2009]Oil 二维前缀和 二维前缀最值

    欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ1177 题意概括 在一个n*m的矩阵中,每一个位置一个数字. 现在让你选出3个k*k的矩阵,它们互不 ...

  6. 前缀和&二维前缀和

    我们知道,数组上的前缀和S[i]=S[i-1]+a[i] 那么,怎样求二维前缀和呢? 二维前缀和: 绿色点的前缀和就是黄色.红色.灰色和绿色的点权和 怎样计算? s[i][j]=s[i-1][j]+s ...

  7. Codeforces 1262E Arson In Berland Forest(二维前缀和+二维差分+二分)

     题意是需要求最大的扩散时间,最后输出的是一开始的火源点,那么我们比较容易想到的是二分找最大值,但是我们在这满足这样的点的时候可以发现,在当前扩散时间k下,以这个点为中心的(2k+1)2的正方形块内必 ...

  8. LeetCode Continuous Subarray Sum 题解 同余前缀和 Hash表

    文章目录 题意 思路 特殊情况k=0 Source Code 1 Source Code 2 题意 给定一个数组和一个整数k,返回是否存在一个长度至少为2的连续子数组的和为k的倍数. 思路 和上一篇博 ...

  9. Codeforces 1093G题解(线段树维护k维空间最大曼哈顿距离)

    题意是,给出n个k维空间下的点,然后q次操作,每次操作要么修改其中一个点的坐标,要么查询下标为[l,r]区间中所有点中两点的最大曼哈顿距离. 思路:参考blog:https://blog.csdn.n ...

随机推荐

  1. 【转】风控中的特征评价指标(三)——KS值

    转自:https://zhuanlan.zhihu.com/p/79934510 风控业务背景 在风控中,我们常用KS指标来评估模型的区分度(discrimination).这也是风控模型同学最为追求 ...

  2. Catalan数以及相关性质的证明

    \(Catalan\) 数相关证明 Mushroom 2021-5-14 \(Catalan\)数的定义 给定一个凸\(n + 1\)边形, 通过在内部不相交的对角线,把它划分成为三角形的组合,不同的 ...

  3. 从effective C++中窥探C++11特性

    这几天在看effective C++3rd,这本书算是比较经典的一本入门C++的书了.虽然年代比较久远书中讲的好多模式已经被的新特性取代了,但是从这些旧的模式中可以了解到一些C++新特性设计的初衷,也 ...

  4. redis分布式锁-可重入锁

    redis分布式锁-可重入锁 上篇redis实现的分布式锁,有一个问题,它不可重入. 所谓不可重入锁,即若当前线程执行某个方法已经获取了该锁,那么在方法中尝试再次获取锁时,就会获取不到被阻塞. 同一个 ...

  5. 搭建LNMP环境部署Wordpress博客

    !!!首先要做的就是关闭系统的防火墙以及selinux: #systemctl stop firewalld #systemctl disable firewalld #sed -ri 's/^(SE ...

  6. RHEL/CentOS 7 中配置 PXE 网络启动服务器

    RHEL/CentOS 7 中配置 PXE 网络启动服务器 作者: Matei Cezar 译者: LCTT joeren | 2015-02-17 14:28   评论: 13 收藏: 8 分享:  ...

  7. 007.Python循环语句while循环嵌套

    1 使用两个循环打印十行小星星 j = 0 while j<10: # 打印一行十个小星星 i = 0 while i<10: print("*",end=" ...

  8. JavaSE 知识图谱

    JAVA基础语法 DOS命令 JAVA介绍 JDK安装 JAVA环境的搭建 关键字 注释 标识符命名规则(编码规范) 字面值常量 进制转换 基本类型 变量(局部变量.静态变量) 运算符 表达式 控制语 ...

  9. MyBatis 映射文件详解(六)

    MyBatis 配置文件类型 MyBatis配置文件有两种类型,如下: 全局配置文件(如 mybatis-config.xml) Mapper XML 映射文件(如 UserMapper.xml) 上 ...

  10. 10.2-3 ifup&ifdown:激活与禁用网络接口

    ifup:激活网络接口     ifup 和 ifdown 命令用于激活指定的网络接口.ifup命令其实是一个Shel脚本,有Shel基础的读者可以使用which命令来找到这个脚本并读一读.命令可读取 ...