[Luogu 2216] [HAOI2007]理想的正方形
[Luogu 2216] [HAOI2007]理想的正方形
题目描述
有一个a*b的整数组成的矩阵,现请你从中找出一个n*n的正方形区域,使得该区域所有数中的最大值和最小值的差最小。
输入输出格式
输入格式:
第一行为3个整数,分别表示a,b,n的值
第二行至第a+1行每行为b个非负整数,表示矩阵中相应位置上的数。每行相邻两数之间用一空格分隔。
输出格式:
仅一个整数,为a*b矩阵中所有“n*n正方形区域中的最大整数和最小整数的差值”的最小值。
输入输出样例
5 4 2
1 2 5 6
0 17 16 0
16 17 2 1
2 10 2 1
1 2 2 2
1
说明
问题规模
(1)矩阵中的所有数都不超过1,000,000,000
(2)20%的数据2<=a,b<=100,n<=a,n<=b,n<=10
(3)100%的数据2<=a,b<=1000,n<=a,n<=b,n<=100
抱着刷DP的心理,打开了这道题,但好像并不会DP qaq,这里介绍一种二维st的方法
题解:
因为本蒟蒻是刚复习了一下st表做RMQ,所以顺手继续做了
因为我们发现n是不变的,所以st表的时候可以只开三维f[a][b][log n]
然后就可以根据一维st表一样的预处理方式,只是一个状态需要从四个状态转移过来
因为一个正方形肯定是可以分成四个部分的,可能包含重叠.
所以就是这样,然后最后查询的时候也是分成四个部分
于是就结束了...
#include<bits/stdc++.h>
using namespace std;
const int N=;
int a,b,n,lg,ans=1e9;
int mp[N][N],f[N][N][],g[N][N][];
int Min(int a,int b,int c,int d){
return min(a,min(b,min(c,d)));
}
int Max(int a,int b,int c,int d){
return max(a,max(b,max(c,d)));
}
int ask1(int x,int y){
int dx=x+n-,dy=y+n-;
return Max(g[x][y][lg],g[x][dy-(<<lg)+][lg],g[dx-(<<lg)+][y][lg],g[dx-(<<lg)+][dy-(<<lg)+][lg]);
}
int ask2(int x,int y){
int dx=x+n-,dy=y+n-;
return Min(f[x][y][lg],f[x][dy-(<<lg)+][lg],f[dx-(<<lg)+][y][lg],f[dx-(<<lg)+][dy-(<<lg)+][lg]);
}
int main(){
scanf("%d%d%d",&a,&b,&n); memset(f,0x3f3f,sizeof(f)); memset(g,,sizeof(g));
for (int i=;i<=a;++i)
for (int j=;j<=b;++j)
scanf("%d",&mp[i][j]),f[i][j][]=g[i][j][]=mp[i][j];
for (int k=;(<<k)<=n;++k)
for (int i=;i+(<<k)-<=a;++i)
for (int j=;j+(<<k)-<=b;++j){
f[i][j][k]=Min(f[i][j][k-],f[i][j+(<<(k-))][k-],f[i+(<<(k-))][j][k-],f[i+(<<(k-))][j+(<<k-)][k-]);
g[i][j][k]=Max(g[i][j][k-],g[i][j+(<<(k-))][k-],g[i+(<<(k-))][j][k-],g[i+(<<(k-))][j+(<<k-)][k-]);
}
lg=(int)(log(n)/log(2.0));
for (int i=;i<=a-n+;++i)
for (int j=;j<=b-n+;++j)
ans=min(ans,ask1(i,j)-ask2(i,j));
printf("%d",ans);
}
然而事实上,我觉得单调队列的做法也是非常好的,于是借鉴了别人的题解,此下贴出
对于每一行,我们维护定长区间内的最大值和最小值,maxv[i][j]表示第i行第j列,从j-k+1~j这些数的最大值,minv[i][j]同理。这里的k是题目中的n,也就是正方形的长。然后我们已经知道每一行定长区间内的最值,对于每一列,我们也同样维护这一列定长区间的最值,就能得到一个“定正方形”内的最值。
至于定长区间的最值怎么求,那就是用到我们的单调队列了,这道题其实是个模板。这里我是开两个双端队列,maxq和minq,分别维护。(当然开一个也可以,那样代码就比较长了)
#include<cstdio>
#include<cstring>
#include<queue>
#include<algorithm>
using namespace std; const int N = ;
const int INF = 1e9;
int n, m, k, a[N][N], maxv[N][N], minv[N][N]; int main()
{
scanf("%d%d%d", &n, &m, &k);
for (int i=; i<=n; i++)
for (int j=; j<=m; j++) scanf("%d", &a[i][j]);
//以下对于每一行用单调队列求出maxv[i][j]和minv[i][j]
for (int i=; i<=n; i++){
deque<int> maxq, minq;
maxv[i][] = ;
minv[i][] = INF;
for (int j=; j<=m; j++){
while (!maxq.empty() && maxq.front() < j-k+) maxq.pop_front(); //如果范围超过k就弹出队列
while (!maxq.empty() && a[i][maxq.back()] <= a[i][j]) maxq.pop_back(); //维护单调递减的队列使得队首为最大值
maxq.push_back(j);
maxv[i][j] = a[i][maxq.front()];
while (!minq.empty() && minq.front() < j-k+) minq.pop_front();
while (!minq.empty() && a[i][minq.back()] >= a[i][j]) minq.pop_back(); //维护单调递增的队列使得队首为最小值
minq.push_back(j);
minv[i][j] = a[i][minq.front()];
}
}
//以下对于每一列用单调队列求出“定正方形”内最值,并直接计算答案
int ans = INF;
for (int j=k; j<=m; j++){ //注意枚举范围从k开始
deque<int> maxq, minq;
int MaxV = ;
int MinV = INF;
for (int i=; i<=n; i++){
//单调队列用法同上
while (!maxq.empty() && maxq.front() < i-k+) maxq.pop_front();
while (!maxq.empty() && maxv[maxq.back()][j] <= maxv[i][j]) maxq.pop_back();
maxq.push_back(i);
MaxV = maxv[maxq.front()][j];
while (!minq.empty() && minq.front() < i-k+) minq.pop_front();
while (!minq.empty() && minv[minq.back()][j] >= minv[i][j]) minq.pop_back();
minq.push_back(i);
MinV = minv[minq.front()][j];
if (i >= k) ans = min(ans, MaxV - MinV); //注意i >= k时才能更新答案
}
}
printf("%d\n", ans);
return ;
}
[Luogu 2216] [HAOI2007]理想的正方形的更多相关文章
- Luogu 2216[HAOI2007]理想的正方形 - 单调队列
Solution 二维单调队列, 这个数组套起来看得我眼瞎... Code #include<cstdio> #include<algorithm> #include<c ...
- Luogu 2216 [HAOI2007]理想的正方形 (单调队列优化)
题意: 给出一个 N×M 的矩阵,以及一个数值 K ,求在给定的矩阵中取出一个 K×K 的矩阵其中最大值减去最小值的最小值. 细节: 没有细节来发暴力走天下,20分也是分啊~~~ QAQ. 分析: 感 ...
- BZOJ1047或洛谷2216 [HAOI2007]理想的正方形
BZOJ原题链接 洛谷原题链接 显然可以用数据结构或\(ST\)表或单调队列来维护最值. 这里采用单调队列来维护. 先用单调队列维护每一行的最大值和最小值,区间长为正方形长度. 再用单调队列维护之前维 ...
- 洛谷 2216 [HAOI2007]理想的正方形
题目戳这里 一句话题意 给你一个a×b的矩形,求一个n×n的子矩阵,矩阵里面的最大值和最小值之差最小. Solution 这个题目许多大佬都是单调队列,但是我不是很会,只好用了比较傻逼的方法: 首先我 ...
- BZOJ1047: [HAOI2007]理想的正方形 [单调队列]
1047: [HAOI2007]理想的正方形 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 2857 Solved: 1560[Submit][St ...
- HAOI2007 理想的正方形
1047: [HAOI2007]理想的正方形 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 1402 Solved: 738[Submit][Sta ...
- RAM——[HAOI2007]理想的正方形
题目:[HAOI2007]理想的正方形 描述: [问题描述] 有一个a*b的整数组成的矩阵,现请你从中找出一个n*n的正方形区域,使得该区域所有数中的最大值和最小值的差最小. [输入]: 第一行为3个 ...
- bzoj 1047 : [HAOI2007]理想的正方形 单调队列dp
题目链接 1047: [HAOI2007]理想的正方形 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 2369 Solved: 1266[Submi ...
- BZOJ 1047: [HAOI2007]理想的正方形( 单调队列 )
单调队列..先对每一行扫一次维护以每个点(x, y)为结尾的长度为n的最大最小值.然后再对每一列扫一次, 在之前的基础上维护(x, y)为结尾的长度为n的最大最小值. 时间复杂度O(ab) (话说还是 ...
随机推荐
- matlab数值数据的表示方法,输出数据以及相关函数
数据类型的分类: 1.整型 无符号整型和带符号整形 带符号整形的最大值是127 >>x=int8(129) 输出结果是x=127 >>x=unit8(129) 输出结果是x=1 ...
- 《LeetCode-0004》 寻找两个有序数组的中位数-Median of Two Sorted Arrays
题目给定两个大小为 m 和 n 的有序数组nums1和 nums2. 请你找出这两个有序数组的中位数,并且要求算法的时间复杂度为 O(log(m + n)). 你可以假设 nums1 和 nums2 ...
- HDU-4055 Number String 动态规划 巧妙的转移
题目链接:https://cn.vjudge.net/problem/HDU-4055 题意 给一个序列相邻元素各个上升下降情况('I'上升'D'下降'?'随便),问有几种满足的排列. 例:ID 答: ...
- VMware Workstation搭建Linux操作系统
1.单击“创建新的虚拟机”选项,并在弹出的“新建虚拟机向导”界面中选择“自定义”单选按钮,然后单击“下一步”. 新建虚拟机向导 2.选择虚拟机硬件兼容性,是否兼容之前旧的版本. 兼容性选择 3.选中“ ...
- 【hiho一下 第144周】机会渺茫
[题目链接]:http://hihocoder.com/contest/hiho144/problem/1 [题意] [题解] 找出两个数相同的因子的个数x 然后两个数各自的因子的个数numa,nub ...
- 清北学堂模拟赛d4t5 b
分析:一眼树形dp题,就是不会写QAQ.树形dp嘛,定义状态肯定有一维是以i为根的子树,其实这道题只需要这一维就可以了.设f[i]为以i为根的子树中的权值和.先处理子树内部的情况,用一个数组son[i ...
- ESXi License过期解决办法
http://blog.sina.com.cn/s/blog_538439270101pqls.html
- OpenCV+iOS开发使用文档
一. 前言 OpenCV是开源的跨平台的计算机视觉库,实现了图像处理.计算机视觉和机器学习的很多通用算法. 对于移动设备没有快速输入的键盘,大的屏幕,其优势在于图像和声音,因此要 ...
- Clojure:解决korma中mysql utf8的问题
当使用korma内置的mysql方法时,无法添加utf-8的支持.解决的方法就是重写mysql方法,代码如下: (defn mysql "改编自korma,添加了utf-8的支持" ...
- swift2.0学习之拓展
拓展:和oc的拓展方法功能差点儿相同.就是给已经存在的类,结构体.枚举,协议类型添加新的方法 拓展语法: 用extensionkeyword声明: extension SomeType { // ne ...