T3118 01完美矩阵【计数,前缀和,差分,好题】
Online Judge:未知
Label:好题,计数,前缀和
题目描述
一个01矩形被称为是完美01矩形,如果满足下面3个条件:
(1)它的四条边上都是1
(2)内部(除了4条边)的0和1的个数之差不超过1
(3)大小至少是2*2
给定一个01矩阵,求可以在其中圈出多少完美01矩形。
输入
第一行两个整数n和m
接下来n行,每行m个数,0或者1.
输出
输出完美01矩形的个数。
样例
Input
4 4
1 1 1 1
1 0 1 1
1 1 0 1
1 1 1 1
5 5
1 0 1 1 1
1 0 1 0 1
1 1 0 1 1
1 0 0 1 1
1 1 1 1 1
Output
3
3
Hint
对于30%的数据,n和m的数据范围\([1,20]\);
对于60%的数据,n和m的数据范围\([1,100]\);
对于100%的数据,n和m的数据范围\([1,300]\).
题解
60pts
\(O(N^4)\)随便怎么敲都可以。直接枚举矩形上的两角。对于随机数据,下面的代码跑的飞快。但是如果有大量1就挂掉了。
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N=302;
int a[N][N],n,m;
int sum[N][N];
int le[N][N],ri[N][N],up[N][N],down[N][N];
inline int read(){
int x=0;char c=getchar();
while(c<'0'||c>'9')c=getchar();
while(c>='0'&&c<='9')x=(x<<3)+(x<<1)+(c^48),c=getchar();
return x;
}
void pre(){
register int i,j;
for(i=1;i<=n;++i){
int lst=0;
for(j=1;j<=m;++j){
if(a[i][j])le[i][j]=++lst;
else lst=0;
}
lst=0;
for(j=m;j>=1;j--){
if(a[i][j])ri[i][j]=++lst;
else lst=0;
}
}
for(j=1;j<=m;++j){
int lst=0;
for(i=1;i<=n;++i){
if(a[i][j])up[i][j]=++lst;
else lst=0;
}
lst=0;
for(i=n;i>=1;i--){
if(a[i][j])down[i][j]=++lst;
else lst=0;
}
}
for(i=1;i<=n;++i)for(j=1;j<=m;++j){
sum[i][j]=sum[i-1][j]+sum[i][j-1]-sum[i-1][j-1]+a[i][j];
}
}
inline int calc(int x,int y,int x2,int y2){
int res=sum[x2][y2]-sum[x-1][y2]-sum[x2][y-1]+sum[x-1][y-1];
return res;
}
namespace p60{
void solve(){
register int i,j,k,l;
ll ans=0;
for(i=1;i<=n;++i)for(j=1;j<=m;++j)if(a[i][j]){
for(k=i+1;k<=i+down[i][j]-1;k++)for(l=j+1;l<=j+ri[i][j]-1;l++)if(a[k][l]){
if((k-up[k][l]+1<=i)&&(l-le[k][l]+1<=j)){
int tmp=calc(i+1,j+1,k-1,l-1),o=(k-i-1)*(l-j-1);
if(abs(o-2*tmp)<=1)++ans;
}
}
}
printf("%lld\n",ans);
}
}
int main(){
// freopen("matrix.in","r",stdin);
// freopen("matrix.out","w",stdout);
n=read(),m=read();
for(register int i=1;i<=n;++i)for(register int j=1;j<=m;++j)a[i][j]=read();
pre();
p60::solve();
}
100pts
看数据范围,正解复杂度应该是\(O(N^3)\)的。
对于这类矩阵的计数题,有一个常见套路做法:
用\(O(N^2)\)时间枚举上下行边界,对于列,在\(O(N)\)时间内扫一遍,用前缀和等维护计数。
step0.预处理
先预处理一个东西s[i][j]
,表示(i,j)的二维前缀和
,但与普通的前缀和不同,在这里将0计为-1,将1计为+1。这样处理有什么好处呢?下面会提到。
for(i=1;i<=n;i++)for(j=1;j<=m;j++){
scanf("%d",&a[i][j]);
s[i][j]=s[i-1][j]+s[i][j-1]-s[i-1][j-1]+(a[i][j]==1?1:-1);
}
这样可以求得以(x,y)
为左上角,(x2,y2)
为右下角的矩形"面积"(为了方便描述,下面都用“面积”表示矩形内的01个数差)。比如这个矩形内0个数比1个数多5,则面积为-5,如果0个数等于1个数,则面积为0。
inline int Area(int x,int y,int x2,int y2){
return s[x2][y2]-s[x-1][y2]-s[x2][y-1]+s[x-1][y-1];
}
step1.统计
先枚举上下边。
for(i=1;i<n;i++)for(j=i+1;j<=n;j++){//上下边
接下里要考虑在\(O(N)\)时间内求得以\(i,j\)为上下边的01完美矩阵个数。
先思考一种比较low的做法。
下面代码中,直接暴力统计左右边在\([k,l]\)范围内的符合条件的矩形个数。虽然看起来是\(O(N^3)\)的,但实际是\(O(N^2)\)的(因为后面\(k=l+1\)跳了一下)。
for(k=1;k<=m;k++)if(a[i][k]&&a[j][k]){
int l=k;
while(l+1<=m&&a[i][l+1]&&a[j][l+1])l++;//找到
if(k==l)continue;
//暴力统计
int o1,o2;
for(o1=k;o1<=l;o1++)for(o2=o1;o2<=l;o2++){
if(是01完美矩形(i,j,o1,o2))计数;
}
k=l+1;
}
瓶颈在于这个\(O(N^2)\)的暴力统计。
解决方法就是用到之前预处理的前缀和。见下面代码。
利用差分思想。每次找到合法的列,查询前面的前缀和是否有和它相差1以内的,计入答案。然后把自己的前缀和加入统计。
for(p=k;p<=l;p++)if(Area(i,p,j,p)==j-i+1){
int pre=Area(i+1,k+1,j-1,p-1)+base,now=Area(i+1,k+1,j-1,p)+base;
ans+=cnt[pre-1]+cnt[pre]+cnt[pre+1];
cnt[now]++;
}
由于矩形面积可能为负,所以要加上一个基量\(base\)。
而且计算完\([k,l]\)内的矩形后,还要清空\(cnt[]\)数组,清空代码如下,和上面反一下:
for(p=k;p<=l;p++)if(Area(i,p,j,p)==j-i+1){
int now=Area(i+1,k+1,j-1,p)+base;
cnt[now]--;
}
综上时间复杂度为\(O(N^3)\),完整代码如下:
//枚举上下边,前缀和统计
#include<bits/stdc++.h>
typedef long long ll;
using namespace std;
const int N=305,base=N*N;
int n,m,a[N][N],s[N][N],cnt[2*N*N];
ll ans=0;
inline int Area(int x,int y,int x2,int y2){
return s[x2][y2]-s[x-1][y2]-s[x2][y-1]+s[x-1][y-1];
}
int main(){
scanf("%d%d",&n,&m);
register int i,j,k,p;
for(i=1;i<=n;i++)for(j=1;j<=m;j++){
scanf("%d",&a[i][j]);
s[i][j]=s[i-1][j]+s[i][j-1]-s[i-1][j-1]+(a[i][j]==1?1:-1);
}
for(i=1;i<n;i++)for(j=i+1;j<=n;j++){//上下边
for(k=1;k<=m;k++)if(a[i][k]&&a[j][k]){
int l=k;
while(l+1<=m&&a[i][l+1]&&a[j][l+1])l++;
if(k==l)continue;
for(p=k;p<=l;p++)if(Area(i,p,j,p)==j-i+1){
int pre=Area(i+1,k+1,j-1,p-1)+base,now=Area(i+1,k+1,j-1,p)+base;
ans+=cnt[pre-1]+cnt[pre]+cnt[pre+1];
cnt[now]++;
}
for(p=k;p<=l;p++)if(Area(i,p,j,p)==j-i+1){
int now=Area(i+1,k+1,j-1,p)+base;
cnt[now]--;
}
k=l+1;
}
}
printf("%lld\n",ans);
}
T3118 01完美矩阵【计数,前缀和,差分,好题】的更多相关文章
- HDU 6336 (规律 + 二维矩阵的前缀和妙用)
题目 给出长度为n 的A矩阵 , 按 int cursor = 0; for (int i = 0; ; ++i) { for (int j = 0; j <= i; ++j) { M[j][i ...
- Codeforces 479E Riding in a Lift:前缀和/差分优化dp
题目链接:http://codeforces.com/problemset/problem/479/E 题意: 有一栋n层的房子. 还有一个无聊的人在玩电梯,每次玩电梯都会从某一层坐到另外一层. 他初 ...
- 北京区域赛I题,Uva7676,A Boring Problem,前缀和差分
转载自https://blog.csdn.net/weixin_37517391/article/details/83821752 题解 其实这题不难,只要想到了前缀和差分就基本OK了. 我们要求的是 ...
- [CSP-S模拟测试]:stone(结论+桶+前缀和+差分)
题目描述 $Cab$有两行石子,每个石子上有一个字母,为$'C''A''B'$中的一个.一开始,在每行第一个石子上站着一只$lucky$,$Cab$每次可以选择一个字母,使得所站石子上字母为该字母的$ ...
- HDU-6514 Monitor(二维前缀和+差分)
http://acm.hdu.edu.cn/showproblem.php?pid=6514 Problem Description Xiaoteng has a large area of land ...
- 【T-SQL基础】01.单表查询-几道sql查询题
概述: 本系列[T-SQL基础]主要是针对T-SQL基础的总结. [T-SQL基础]01.单表查询-几道sql查询题 [T-SQL基础]02.联接查询 [T-SQL基础]03.子查询 [T-SQL基础 ...
- luogu 1360 阵容均衡(前缀和+差分+hash)
要求一段最大的区间里每个能力的增长值是一样的. 我们首先求一遍前缀和,发现,如果区间内[l,r]每个能力的增长值是一样的话,那么前缀和[r]和[l-1]的差分也应该是一样的. 那么我们把前缀和的差分h ...
- CF 816B Karen and Coffee【前缀和/差分】
To stay woke and attentive during classes, Karen needs some coffee! Karen, a coffee aficionado, want ...
- HihoCoder1673 : 01间隔矩阵([Offer收割]编程练习赛41)(单调队列)
描述 给定一个N × M的01矩阵,小Hi希望从中找到一个01间隔的子矩阵,并且子矩阵的面积越大越好. 例如对于 0101010 1000101 0101010 1010101 0101010 在右侧 ...
随机推荐
- NX二次开发-NXString转换为char*方法
NX9+VS2012 #include <uf.h> #include <uf_drf.h> #include <NXOpen/Annotations_Note.hxx& ...
- Portainer Exec Container 失败解决方案
近日,将portainer服务挂了个域名,然后用Nginx代理的时候发现不能attach容器了,经过搜索在issue 找到解决方案 1.修改Nginx config server { listen 8 ...
- [转] .htaccess实现www 与没有www之间的重定向
建站过程中有时候我们需要做这些设置 1.访问www 直接重定向到没有www上或者反过来,那么怎么通过.htaccess文件来实现呢. 1.首先服务器要支持Rewrite重写 2.创建.htaccess ...
- JVM内核-原理、诊断与优化学习笔记(四):GC算法与种类
文章目录 GC的概念 GC算法 引用计数法 引用计数法的问题 标记清除 标记压缩 小问题 复制算法 复制算法的最大问题是:空间浪费 整合标记清理思想 -XX:+PrintGCDetails的输出 gc ...
- Beaglebone Black的引脚分配
转自:http://blog.csdn.net/daxueba/article/details/50998000 Beaglebone Black的引脚分配 绝大多数的微型开发平台都提供了一些称为GP ...
- Mysql 插入数据,随机事件选择
在拼写sql的 时候,mysql字段如果需要添加当前时间可以用NOW() 函数 // String sql = ("insert into tablename(content, create ...
- Python: 比较两个字典是否相等
有些情况下会遇到比较两个字典是否相等的问题 直观来想,会比较键是否一致,其对应的值是否相等 python中,还有有另外两种方法: 直接使用== a = {'a': 1, 'b': 2} b = {'a ...
- 科普帖:深度学习中GPU和显存分析
知乎的一篇文章: https://zhuanlan.zhihu.com/p/31558973 关于如何使用nvidia-smi查看显存与GPU使用情况,参考如下链接: https://blog.csd ...
- 与960 Grid System相关的那些问题
为什么是960px? 一直以来,网页设计师都希望寻找一个理想的页面宽度值,既能适应大部分屏幕,又尽可能的在一行显示更多的信息. 我们首先会考虑的是全屏自适应,但这并非一个好的解决方案.一方面,需要做一 ...
- CSS 命名规范将省下调试时间
我听说很多开发者厌恶 CSS.而在我的经验中,这往往是由于他们并没有花时间来学习 CSS. CSS 算不上是最优美的『语言』,但迄今二十多年来,它都是美化 web 举足轻重的工具.从这点来说,也还算不 ...