【bzoj1047】理想的正方形

题意

给定\(a*b\)由整数组成的矩形。

现请你从中找出一个n*n的正方形区域,使得该区域所有数中的最大值和最小值

的差最小。

\(1\leq a,b\leq 1000\)

\(1\leq n\leq 100\)

分析

枚举每一个位置,然后考虑快速求矩形内的最大值和最小值即可。

单调队列可以快速实现:

先求出\(d[i][j]\)表示\(a[i][j-n+1,j-n,...,j]\)中的最值。

然后求出\(f[i][j]\)表示\(d[i-n+1,i-n,...,i][j]\)中的最值。

所有的\(f[i][j]\)就表示以\((i,j)\)为右下角端点的矩形的最值。

也可以使用ST表。

由于\(n\)一定,所以只需要用一个简化的二维ST表即可。

\(f[i][j][k]\)表示跨度为\(2^i\),终点在\((j,k)\)的矩形的最值。

注意ST表的正确姿势。

对于二维ST表,记\(f[i][j][k][l]\),其中两个跨度放前面。

不然调试起来会很麻烦的。

代码

#include <cstdio>
#include <cctype>
#include <cmath>
#include <climits>
#include <algorithm>
using namespace std;

#define rep(i,a,b) for (int i=(a);i<=(b);i++)

const int N=1001;
const int U=10;

const int MAX=INT_MAX>>1;
const int MIN=INT_MIN>>1;

int n,m,siz;
int a[N][N];

int un,um; int unit;
int maxV[U][N][N],minV[U][N][N];
int res;

int rd(void) {
    int x=0,f=1; char c=getchar();
    for (;!isdigit(c);c=getchar()) if (c=='-') f=-1;
    for (;isdigit(c);c=getchar()) x=x*10+c-'0';
    return x*f;
}

int Query(int x,int y) {
    int tx=(x-siz+1)+(1<<unit)-1;
    int ty=(y-siz+1)+(1<<unit)-1;

    int mx=MIN;
    mx=max(mx,maxV[unit][x][y]);
    mx=max(mx,maxV[unit][x][ty]);
    mx=max(mx,maxV[unit][tx][y]);
    mx=max(mx,maxV[unit][tx][ty]);

    int mn=MAX;
    mn=min(mn,minV[unit][x][y]);
    mn=min(mn,minV[unit][x][ty]);
    mn=min(mn,minV[unit][tx][y]);
    mn=min(mn,minV[unit][tx][ty]);

    return mx-mn;
}

int main(void) {
    #ifndef ONLINE_JUDGE
    freopen("bzoj1047.in","r",stdin);
    freopen("bzoj1047.out","w",stdout);
    #endif

    n=rd(),m=rd(),siz=rd();
    rep(i,1,n) rep(j,1,m)
        a[i][j]=rd();

    unit=(int)(log(siz)/log(2));
    rep(i,0,unit) rep(j,1,n) rep(k,1,m) {
        minV[i][j][k]=MAX;
        maxV[i][j][k]=MIN;
    }
    rep(j,1,n) rep(k,1,m) {
        minV[0][j][k]=a[j][k];
        maxV[0][j][k]=a[j][k];
    }
    rep(i,1,unit) rep(j,1,n) rep(k,1,m) {
        int tj=max(1,j-(1<<(i-1)));
        int tk=max(1,k-(1<<(i-1)));

        int *now=&(minV[i][j][k]);
        *now=min(*now,minV[i-1][j][k]);
        *now=min(*now,minV[i-1][j][tk]);
        *now=min(*now,minV[i-1][tj][k]);
        *now=min(*now,minV[i-1][tj][tk]);

        now=&(maxV[i][j][k]);
        *now=max(*now,maxV[i-1][j][k]);
        *now=max(*now,maxV[i-1][j][tk]);
        *now=max(*now,maxV[i-1][tj][k]);
        *now=max(*now,maxV[i-1][tj][tk]);
    }

    res=MAX;
    rep(i,siz,n) rep(j,siz,m) {
        int t=Query(i,j);
        res=min(res,t);
    }
    printf("%d\n",res);

    return 0;
}

【bzoj1047】理想的正方形的更多相关文章

  1. bzoj1047理想的正方形

    题目链接 纪念又双叒叕的一道暴力碾标算的题 我们考虑纯暴力 #include<iostream> #include<cstdio> #include<algorithm& ...

  2. bzoj1047 理想的正方形

    Description 有一个a*b的整数组成的矩阵,现请你从中找出一个n*n的正方形区域,使得该区域所有数中的最大值和最小值的差最小. Input 第一行为3个整数,分别表示a,b,n的值第二行至第 ...

  3. 【BZOJ1047】[HAOI2007]理想的正方形(单调队列,动态规划)

    [BZOJ1047][HAOI2007]理想的正方形(单调队列,动态规划) 题面 BZOJ 洛谷 题解 直接一个单调队列维护一下没给点和它前面的\(n\)个位置的最大值,再用一次单调队列维护连续\(n ...

  4. 【BZOJ1047】[HAOI2007]理想的正方形

    [BZOJ1047][HAOI2007]理想的正方形 题面 bzoj 洛谷 题解 二维\(st\)表,代码是以前的 #include<iostream> #include<cstdi ...

  5. [bzoj1047][HAOI2007]理想的正方形_动态规划_单调队列

    理想的正方形 bzoj-1047 HAOI-2007 题目大意:有一个a*b的整数组成的矩阵,现请你从中找出一个n*n的正方形区域,使得该区域所有数中的最大值和最小值的差最小. 注释:$2\le a, ...

  6. BZOJ1047: [HAOI2007]理想的正方形 [单调队列]

    1047: [HAOI2007]理想的正方形 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 2857  Solved: 1560[Submit][St ...

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

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

  8. 【BZOJ1047】[HAOI2007]理想的正方形 (倍增ST表)

    [HAOI2007]理想的正方形 题目描述 有一个\(a*b\)的整数组成的矩阵,现请你从中找出一个\(n*n\)的正方形区域,使得该区域所有数中的最大值和最小值的差最小. 输入输出格式 输入格式: ...

  9. HAOI2007 理想的正方形

    1047: [HAOI2007]理想的正方形 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1402  Solved: 738[Submit][Sta ...

随机推荐

  1. MUI 微信 和支付宝支付 (前台代码)

    <!-- 校园公告详情界面 用于显示校园公告的详情信息 在校园公告界面点击某一条目后 进入本界面查看详情 --> <!DOCTYPE html> <html> &l ...

  2. mfc线程

    1.生成线程 方式1. HANDLE hthread; //线程句柄 hthread=CreateThread(NULL,0,(LPTHREAD_START_ROUTINE)threadFunc,NU ...

  3. swift + jj实践

    1,UIButton里面的字体和图片左对齐 button.imageEdgeInsets = UIEdgeInsetsMake(0,180/2,0.0,320/2) let btnRect = but ...

  4. 查询mysql数据库中所有用户及用户权限

    SELECT DISTINCT CONCAT('User: ''',user,'''@''',host,''';') AS query FROM mysql.user;

  5. 在Linux命令行下令人惊叹的惊叹号(!)

    '!'符号在Linux中不但可以用作否定符号,还可以用来从历史命令记录中取出命令或不加修改的执行之前运行的命令.下面的所有命令都已经在Bash Shell中经过确切地检验.尽管我没有试过,但大多都不能 ...

  6. 【Unity3D游戏开发】性能优化之spine提高80~90%的效率 (三一)

    Spine效率低 Unity项目加载spine动画,经常会出现卡顿的情况,如游戏中瞬间播放一个动画,打开一个带spine动画的界面.尤其是SkeletonRenderer.Awake时,会瞬间出现大量 ...

  7. MSChart使用

    制作报表的时候结果出现画红线处的信息太散, 如果没必要全部显示出来,我们可以使用这种效果: 注意和前面的区分,这个功能叫做Collect Pie Slices(收集分区) 要实现此功能,应先了解相关信 ...

  8. linux设置tomcat开机自启动

    本文假设jdk环境安装成功,如何安装JDK请参考这个链接: http://www.cnblogs.com/yoyotl/p/5395208.html 1. 下载apache的安装包,例如本例下载了ap ...

  9. wooyunAPI

    经常要爬去乌云的信息,但是每次都是硬爬,写完了发现乌云有提供API的,整理给大家: 1. WooYun Api是什么 通过WooYun开放的Api接口,其它网站或应用可以根据自己获取的权限调用WooY ...

  10. CUBRID学习笔记 35 net驱动错误码和信息 cubrid教程示例

    DO.NET Error Code Number Error Code Error Message Note 0 ER_NO_ERROR "No Error"   1 ER_NOT ...