Description

有一个a*b的整数组成的矩阵,现请你从中找出一个n*n的正方形区域,使得该区域所有数中的最大值和最小值的差最小。
Input

第一行为3个整数,分别表示a,b,n的值第二行至第a+1行每行为b个非负整数,表示矩阵中相应位置上的数。每行相邻两数之间用一空格分隔。
Output

仅一个整数,为a*b矩阵中所有“n*n正方形区域中的最大整数和最小整数的差值”的最小值。
Sample Input
5 4 2
1 2 5 6
0 17 16 0
16 17 2 1
2 10 2 1
1 2 2 2
Sample Output
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

看了题解,过了好几天又看这道题,感慨万分,好久不做RMQ果然什么都忘了

就是先做一遍右边的,把右边n格的最大值最小值存到最左边这一格

然后做一遍向下的,把下面n格的最值存到这一格,现在,每一个n*n的正方形的信息都存到左上角了,扫一遍就行了

 const
maxn=;
var
a,b:array[..maxn,..maxn]of longint;
n,m,k:longint; procedure down(var x:longint;y:longint);
begin
if x>y then x:=y;
end; procedure up(var x,y:longint);
begin
if x<y then x:=y;
end; procedure main;
var
i,j,l,s,ans:longint;
begin
read(n,m,k);
for i:= to n do
for j:= to m do
begin
read(a[i,j]);
b[i,j]:=a[i,j];
end;
l:=;
while l<k do
begin
for i:= to n do
for j:= to m do
begin
s:=j+l;
down(s,m-l+);
down(s,j+k-l);
up(a[i,j],a[i,s]);
down(b[i,j],b[i,s]);
end;
l:=l<<;
end;
l:=;
while l<k do
begin
for i:= to n do
for j:= to m do
begin
s:=i+l;
down(s,n-l+);
down(s,i+k-l);
up(a[i,j],a[s,j]);
down(b[i,j],b[s,j]);
end;
l:=l<<;
end;
ans:=maxlongint;
for i:= to n-k+ do
for j:= to m-k+ do
down(ans,a[i,j]-b[i,j]);
write(ans);
end; begin
main;
end.

结果写完就被鄙视了,z55250825 ORZ,其实可以用单调队列

 const
maxn=;
var
a,b:array[..maxn,..maxn]of longint;
n,m,k,ans:longint; function min(x,y:longint):longint;
begin
if x<y then exit(x);
exit(y);
end; procedure init;
var
i,j:longint;
begin
read(n,m,k);
for i:= to n do
for j:= to m do
begin
read(a[i,j]);
b[i,j]:=a[i,j];
end;
end; var
q:array[..maxn,..]of longint;
head,tail:longint; procedure work;
var
i,j:longint;
begin
for i:= to n do
begin
head:=;
tail:=;
for j:= to m do
begin
while (tail>=head) and (q[tail,]<=a[i,j]) do
dec(tail);
inc(tail);
q[tail,]:=j;
q[tail,]:=a[i,j];
while q[head,]<=j-k do
inc(head);
a[i,j]:=q[head,];
end;
end;
for j:= to m do
begin
head:=;
tail:=;
for i:= to n do
begin
while (tail>=head) and (q[tail,]<=a[i,j]) do
dec(tail);
inc(tail);
q[tail,]:=i;
q[tail,]:=a[i,j];
while q[head,]<=i-k do
inc(head);
a[i,j]:=q[head,];
end;
end;
for i:= to n do
begin
head:=;
tail:=;
for j:= to m do
begin
while (tail>=head) and (q[tail,]>=b[i,j]) do
dec(tail);
inc(tail);
q[tail,]:=j;
q[tail,]:=b[i,j];
while q[head,]<=j-k do
inc(head);
b[i,j]:=q[head,];
end;
end;
for j:= to m do
begin
head:=;
tail:=;
for i:= to n do
begin
while (tail>=head) and (q[tail,]>=b[i,j]) do
dec(tail);
inc(tail);
q[tail,]:=i;
q[tail,]:=b[i,j];
while q[head,]<=i-k do
inc(head);
b[i,j]:=q[head,];
end;
end;
ans:=maxlongint;
for i:=k to n do
for j:=k to m do
ans:=min(ans,a[i,j]-b[i,j]);
write(ans);
end; begin
init;
work;
end.

Wikioi上实在是过不了,然后写了一个C++的

 #include<cstdio>
using namespace std; const int maxn=; int a[maxn][maxn],b[maxn][maxn],n,m,k,ans; int min(int x,int y)
{
return(x<y?x:y);
} void init()
{
int i,j;
scanf("%d%d%d",&n,&m,&k);
for(i=;i<=n;i++)
for(j=;j<=m;j++)
scanf("%d",&a[i][j]),b[i][j]=a[i][j];
} int q[maxn][],head,tail; void work()
{
int i,j;
for(i=;i<=n;i++)
{
head=;tail=;
for(j=;j<=m;j++)
{
while(tail>=head & q[tail][]<=a[i][j])--tail;
tail++;
q[tail][]=j;
q[tail][]=a[i][j];
while(q[head][]<=j-k)head++;
a[i][j]=q[head][];
}
}
for(j=;j<=m;j++)
{
head=;tail=;
for(i=;i<=n;i++)
{
while(tail>=head & q[tail][]<=a[i][j])tail--;
tail++;
q[tail][]=i;
q[tail][]=a[i][j];
while(q[head][]<=i-k)head++;
a[i][j]=q[head][];
}
}
for(i=;i<=n;i++)
{
head=;tail=;
for(j=;j<=m;j++)
{
while(tail>=head & q[tail][]>=b[i][j])tail--;
tail++;
q[tail][]=j;
q[tail][]=b[i][j];
while(q[head][]<=j-k)head++;
b[i][j]=q[head][];
}
}
for(j=;j<=m;j++)
{
head=;tail=;
for(i=;i<=n;i++)
{
while(tail>=head & q[tail][]>=b[i][j])tail--;
tail++;
q[tail][]=i;
q[tail][]=b[i][j];
while(q[head][]<=i-k)head++;
b[i][j]=q[head][];
}
}
ans=;
for(i=k;i<=n;i++)
for(j=k;j<=m;j++)
ans=min(ans,a[i][j]-b[i][j]);
printf("%d",ans);
} int main()
{
init();
work();
return ;
}

1047: [HAOI2007]理想的正方形 - BZOJ的更多相关文章

  1. bzoj 1047 : [HAOI2007]理想的正方形 单调队列dp

    题目链接 1047: [HAOI2007]理想的正方形 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 2369  Solved: 1266[Submi ...

  2. BZOJ 1047: [HAOI2007]理想的正方形( 单调队列 )

    单调队列..先对每一行扫一次维护以每个点(x, y)为结尾的长度为n的最大最小值.然后再对每一列扫一次, 在之前的基础上维护(x, y)为结尾的长度为n的最大最小值. 时间复杂度O(ab) (话说还是 ...

  3. [BZOJ 1047] [HAOI2007] 理想的正方形 【单调队列】

    题目链接:BZOJ - 1047 题目分析 使用单调队列在 O(n^2) 的时间内求出每个 n * n 正方形的最大值,最小值.然后就可以直接统计答案了. 横向有 a 个单调队列(代码中是 Q[1] ...

  4. 【BZOJ】1047: [HAOI2007]理想的正方形(单调队列/~二维rmq+树状数组套树状数组)

    http://www.lydsy.com/JudgeOnline/problem.php?id=1047 树状数组套树状数组真心没用QAQ....首先它不能修改..而不修改的可以用单调队列做掉,而且更 ...

  5. BZOJ 1047: [HAOI2007]理想的正方形

    题目 单调队列是个很神奇的东西,我以前在博客写过(吧) 我很佩服rank里那些排前几的大神,700ms做了时限10s的题,简直不能忍.(但是我还是不会写 我大概一年半没写单调队列,也有可能根本没有写过 ...

  6. BZOJ 1047: [HAOI2007]理想的正方形 单调队列瞎搞

    题意很简明吧? 枚举的矩形下边界和右端点即右下角,来确定矩形位置: 每一个纵列开一个单调队列,记录从 i-n+1 行到 i 行每列的最大值和最小值,矩形下边界向下推移的时候维护一下: 然后在记录的每一 ...

  7. bzoj 1047: [HAOI2007]理想的正方形【单调队列】

    没有复杂结构甚至不长但是写起来就很想死的代码类型 原理非常简单,就是用先用单调队列处理出mn1[i][j]表示i行的j到j+k-1列的最小值,mx1[i][j]表示i行的j到j+k-1列的最大值 然后 ...

  8. 1047: [HAOI2007]理想的正方形

    Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 4075  Solved: 2277[Submit][Status][Discuss] Descript ...

  9. bzoj千题计划215:bzoj1047: [HAOI2007]理想的正方形

    http://www.lydsy.com/JudgeOnline/problem.php?id=1047 先用单调队列求出每横着n个最大值 再在里面用单调队列求出每竖着n个的最大值 这样一个位置就代表 ...

随机推荐

  1. C# 截图类

    注意修改命名空间using System.Drawing; using System.Windows.Forms;   namespace WindowsFormsApplication1 {     ...

  2. Linux 命令 - arp: 操作系统的 ARP 缓存

    arp 命令可以查看 ARP 缓存或者手动添加.删除缓存中的条目. 命令格式 arp [-evn] [-H type] [-i if] -a [hostname] arp [-v] [-i if] - ...

  3. HTTP - 摘要认证

    基本认证便捷灵活,但极不安全.用户名和密码都是以明文形式传送的,也没有采取任何措施防止对报文的篡改.安全使用基本认证的唯一方式就是将其与 SSL 配合使用. 摘要认证是另一种 HTTP 认证协议,它与 ...

  4. c#中网络异常的处理办法

    加入try catch来判断,catch使用的WebException来处理 try { var request = WebRequest.Create(uri); using (var respon ...

  5. ASP.NET网站前端页面的复制

    网络普及的时代,遇到问题的首要解决方案并不是问人,而是找度娘.当我们找一些技术性的问题时,会发现很多解决方案在博客里,看看博主发表的博客总是惊叹不已,想要自己也有这么一个好习惯,把学到的东西以自己的方 ...

  6. 软件包 java.util 的分层结构

    概述  软件包  类  使用   树  已过时  索引  帮助  JavaTM Platform Standard Ed. 6  上一个   下一个 框架    无框架    所有类         ...

  7. Android手机播放电脑视频文件-屌丝必备

    今天早上一到办公室,照常打开博客园看文章,看到有一片文章是用  http://www.cnblogs.com/wdfrog/p/3738180.html 看到这哥们实现的方法好复杂,又是配置电脑端,又 ...

  8. Android Studio ndk-Jni开发详细

    http://www.open-open.com/lib/view/open1451917048573.html Java Native Interface (JNI)标准是java平台的一部分,它允 ...

  9. Sublime Text 3下 Emmet 使用小技巧

    Emmet常用技巧:(输入下面简写,按Tab键可触发效果)                          生成 HTML 文档初始结构 html:5 或者 ! 生成 HTML5 结构        ...

  10. 【转帖】error C2296: “^”: 非法,左操作数包含“double”类型

    想要实现 ,写的C++程序为 double x; x=2^3; 结果程序总是出现这样的错误:error C2296: “^”: 非法,左操作数包含“double”类型 后来才发现操作符“^”,在C++ ...