【二维单调队列】BZOJ1047-[HAOI2007]理想的正方形
【题目大意】
有一个a*b的整数组成的矩阵,现请你从中找出一个n*n的正方形区域,使得该区域所有数中的最大值和最小值的差最小。
【思路】
裸的二维单调队列。二维单调队列的思路其实很简单:
(1)对于每一行维护两个宽度为n的滑动窗口记录单行中的min和max,和POJ2823一个道理。此时相当于把n个格子浓缩到了一个格子当中。
(2)维护n*n大小的二维滑动窗口中的min和max。由于有了第一步操作,只要考虑每一个n*n的矩形右上角到右下角的最值即可。相当于对于每一列,维护两个宽度为n的滑动窗口。此时循环要行和列里外颠倒,而列只要从第n列开始维护即可(我一开始就错在了这两个地方)
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define INF 0x7fffffff
struct node
{
int data,pos;
};
using namespace std;
const int MAXN=+;
int a,b,n;
int num[MAXN][MAXN];
int rmin[MAXN][MAXN],rmax[MAXN][MAXN],smin[MAXN][MAXN],smax[MAXN][MAXN]; void init()
{
scanf("%d%d%d",&a,&b,&n);
for (int i=;i<a;i++)
for (int j=;j<b;j++) scanf("%d",&num[i][j]);
} void humdrum_queue()
{
/*对于每一行,用单调队列维护[x-n+1,x]的最小值和最大值*/
for (int i=;i<a;i++)
{
int minhead=,mintail=,maxhead=,maxtail=;
node qmin[MAXN],qmax[MAXN];
for (int j=;j<b;j++)
{
int nown=num[i][j];
/*维护最小值*/
while (minhead<mintail && qmin[mintail-].data>nown) mintail--;
qmin[mintail++]=(node){nown,j};
while (minhead<mintail && qmin[minhead].pos<=j-n) minhead++;
rmin[i][j]=qmin[minhead].data; /*维护最大值*/
while (maxhead<maxtail && qmax[maxtail-].data<nown) maxtail--;
qmax[maxtail++]=(node){nown,j};
while (maxhead<maxtail && qmax[maxhead].pos<=j-n) maxhead++;
rmax[i][j]=qmax[maxhead].data;
}
} /*对于以[i,j]为右下角的n*n矩形,用单调队列维护它的最小值和最大值
这步操作可以用单调队列维护每一列,相当于维护该矩形右上角到右下角对应点的rmin与rmax值*/
/*|ATTENTION|要注意的是这里i和j要颠倒过来,所以内外循环以及队列中的pos均要颠倒!*/
for (int j=n-;j<b;j++)
{
int minhead=,mintail=,maxhead=,maxtail=;
node qmin[MAXN],qmax[MAXN];
for (int i=;i<a;i++)
{
int nowmin=rmin[i][j],nowmax=rmax[i][j];
/*维护最小值*/
while (minhead<mintail && qmin[mintail-].data>nowmin) mintail--;
qmin[mintail++]=(node){nowmin,i};
while (minhead<mintail && qmin[minhead].pos<=i-n) minhead++;
smin[i][j]=qmin[minhead].data; /*维护最大值*/
while (maxhead<maxtail && qmax[maxtail-].data<nowmax) maxtail--;
qmax[maxtail++]=(node){nowmax,i};
while (maxhead<maxtail && qmax[maxhead].pos<=i-n) maxhead++;
smax[i][j]=qmax[maxhead].data;
}
}
} void getans()
{
int ans=INF;
for (int i=n-;i<a;i++)
for (int j=n-;j<b;j++)
if (smax[i][j]-smin[i][j]<ans) ans=smax[i][j]-smin[i][j];
cout<<ans<<endl;
} int main()
{
init();
humdrum_queue();
getans();
return ;
}
【二维单调队列】BZOJ1047-[HAOI2007]理想的正方形的更多相关文章
- bzoj1047-理想的正方形(二维单调队列)
题意: 给一个矩阵,给出行列和每个数,再给出一个N,求出所有N*N的子矩阵中最大值最小值之差的最小值解析: 暴力枚举肯定不行,这题可以用二维单调队列做,把同一行的连续N个点缩成一个点保存最大最小值预处 ...
- BZOJ1047: [HAOI2007]理想的正方形 [单调队列]
1047: [HAOI2007]理想的正方形 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 2857 Solved: 1560[Submit][St ...
- bzoj千题计划215:bzoj1047: [HAOI2007]理想的正方形
http://www.lydsy.com/JudgeOnline/problem.php?id=1047 先用单调队列求出每横着n个最大值 再在里面用单调队列求出每竖着n个的最大值 这样一个位置就代表 ...
- [BZOJ1047][HAOI2007]理想的正方形 二维单调队列
题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=1047 我们对每矩阵的一列维护一个大小为$n$的单调队列,队中元素为矩阵中元素.然后扫描每一 ...
- bzoj1047 [HAOI2007]理想的正方形——二维单调队列
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1047 就是先对行做一遍单调队列,再对那个结果按列做一遍单调队列即可. 代码如下: #incl ...
- [luoguP2216] [HAOI2007]理想的正方形(二维单调队列)
传送门 1.先弄个单调队列求出每一行的区间为n的最大值最小值. 2.然后再搞个单调队列求1所求出的结果的区间为n的最大值最小值 3.最后扫一遍就行 懒得画图,自己体会吧. ——代码 #include ...
- [bzoj1047][HAOI2007]理想的正方形_动态规划_单调队列
理想的正方形 bzoj-1047 HAOI-2007 题目大意:有一个a*b的整数组成的矩阵,现请你从中找出一个n*n的正方形区域,使得该区域所有数中的最大值和最小值的差最小. 注释:$2\le a, ...
- BZOJ1047[HAOI2007]理想的正方形——二维ST表
题目描述 有一个a*b的整数组成的矩阵,现请你从中找出一个n*n的正方形区域,使得该区域所有数中的最大值和最小值的差最小. 输入 第一行为3个整数,分别表示a,b,n的值第二行至第a+1行每行为b个非 ...
- 【单调队列】bzoj1047 [HAOI2007]理想的正方形
先把整个矩阵处理成b[n][m-K+1].c[n][m-K+1]大小的两个矩阵,分别存储每行每K个数中的最大.最小值,然后再通过b.c处理出d.e分别表示K*K大小的子矩阵中的最大.最小值即可.单调队 ...
随机推荐
- POJ3020:Antenna Placement(二分图匹配)
Antnna Placement Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 11093 Accepted: 5459 ...
- HDU4185:Oil Skimming(二分图最大匹配)
Oil Skimming Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Tota ...
- 桥接物理网卡,pipwork指定ip,外网连接,研究salt+docker
1.桥接物理网卡: 首先下载工具: yum -y install --enablerepo=epel bridge-utils 停止服务: 983 systemctl stop docker 删除do ...
- 接口认证方式:Bearer Token
因为HTTP协议是开放的,可以任人调用.所以,如果接口不希望被随意调用,就需要做访问权限的控制,认证是好的用户,才允许调用API. 目前主流的访问权限控制/认证模式有以下几种: 1),Bearer T ...
- 前端面试:css预处理
css预处理定义: 定义了一种新的语言,其基本思想是用一种专门编程语言,为css增加了一些编程的特性,将css作为目标生成文件,然后开发者就只要使用这种语言进行编码工作. 几种预处理语言 sass l ...
- 关于fragment点击能穿透问题
本人在做项目的过程中遇到的这个问题,然后就在网上百度了一下,之后也是在csdn上看到博友发过此类问题的解决办法,所以特此重新总结一下,顺便也给自己提个醒,避免出现此类问题.好!下面我们说一下问题: 举 ...
- Maven环境搭建、调试、打包
1.配置Maven环境 将下载文件解压,然后设置maven环境 新建环境变量M2_HOME 变量名:M2_HOME 变量值:F:\maven\apache-maven-3.0.3 追加path环境变量 ...
- 【比赛】百度之星2017 初赛Round A
第一题 题意:给定多组数据P,每次询问P进制下,有多少数字B满足条件:只要数位之和是B的倍数,该数字就是B的倍数. 题解:此题是参考10进制下3和9倍数的特殊性质. 对于10进制,ab=10*a+b= ...
- swift出师作,史丹佛大学游戏制作案例,计算器,小游戏
这两个案例得好好弄清楚,感觉在任何方面既然能够作为公开课被提到这所名校的课程里面自然有不得不学习的理由,感觉应该去入手一下,毕竟这种课,价格不匪,难以接触,能看到就当再教育了.
- bzoj 2151 贪心
几乎完全类似于1150的思路,直接参考那个就行了. http://www.cnblogs.com/BLADEVIL/p/3527193.html /************************** ...