hdu 2888 Check Corners

/*
题意:给出m,n代表给出长为m宽为n的矩阵,然后给出一个q代表查询的次数,然后q行每行给出一个 r1, c1, r2, c2 且满足(1 <= r1 <= r2 <= m, 1 <= c1 <= c2 <= n)
输出这个矩形范围内的最大值,如果最大值是四个顶点某一个的话就输出yes否则输出no
*/
#include<stdio.h>
#include<math.h>
#define Max(a,b) a>b?a:b
int dp[305][305][9][9];//dp[x][y][i][j]dp状态的定义是前两维代表的是横纵坐标,后两维i代表的是在横坐标上2^i,j代表的是纵坐标上2^j
int getMax(int a,int b,int c,int d)
{
    int Mx=(int)(log(c-a+1)/log(2));
    int My=(int)(log(d-b+1)/log(2));
    return Max((Max(dp[a][b][Mx][My],dp[c-(1<<Mx)+1][b][Mx][My])),(Max(dp[a][d-(1<<My)+1][Mx][My],dp[c-(1<<Mx)+1][d-(1<<My)+1][Mx][My])));
}
int dd(int a,int b,int c,int d,int Maax)
{
    if(dp[a][b][0][0]==Maax)
        return 1;
    if(dp[a][d][0][0]==Maax)
        return 1;
    if(dp[c][b][0][0]==Maax)
        return 1;
    if(dp[c][d][0][0]==Maax)
        return 1;
    return 0;
}
int main()
{
    int m,n;
    while(scanf("%d%d",&n,&m)!=EOF)
    {
        for(int i=1; i<=n; i++)
            for(int j=1; j<=m; j++)
                scanf("%d",&dp[i][j][0][0]);//当i==j==0时dp[x][y][0][0]存的是点的值
        int Mx=(int)(log(n)/log(2));//横坐标的倍增最大范围
        int My=(int)(log(m)/log(2));//纵坐标的倍增最大范围
        for(int i=0; i<=Mx; i++)
        for(int j=0; j<=My; j++)
        {
            if(i==0&&j==0)//当i==j==0时这个是一个点
                    continue;
                for(int x=1; x+(1<<i)-1<=n; x++)//把下面的两层for循环整体看的话就是遍历一遍原来的二维地图
                    for(int y=1; y+(1<<j)-1<=m; y++)
                    {
                        if(i==0) dp[x][y][i][j]=Max(dp[x][y][i][j-1],dp[x][y+(1<<(j-1))][i][j-1]);//当i==0时说明i控制的1的范围也就是,在每一列上进行dp相当于原来的一维rmq
                        else     dp[x][y][i][j]=Max(dp[x][y][i-1][j],dp[x+(1<<(i-1))][y][i-1][j]);/*当i!=0的时候说明每一列的dp已经知道
                        我们把行进行分开进行dp,可以整体考虑为,以x,y为起点的在行的宽为2^i,纵宽为2^j这样矩形的最大值
                        */
                    }
        }
        scanf("%d",&m);
        int a,b,c,d;
        for(int i=0;i<m;i++)
        {
            scanf("%d%d%d%d",&a,&b,&c,&d);
            int Maax=getMax(a,b,c,d);
            printf("%d %s\n",Maax,dd(a,b,c,d,Maax)?"yes":"no");
        }
    }
    return 0;
}
/*
10 10
1 2 3 4 5 6 7 8 9 10
1 2 3 4 5 6 7 8 9 10
1 2 3 4 5 6 7 8 9 10
1 2 3 4 5 6 7 8 9 10
1 2 3 4 5 6 7 8 9 10
1 2 3 4 5 6 7 8 9 10
1 2 3 4 5 6 7 8 9 10
1 2 3 4 5 6 7 8 9 10
1 2 3 4 5 6 7 8 9 10
2 3 4 5 6 7 8 9 10 1
100
10 1 10 10
*/

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值