HDU 2888 Check Corners(簡單二維RMQ)
http://acm.hdu.edu.cn/showproblem.php?pid=2888
題意:
給定一個n * m的矩陣,再給定q個詢問,每次詢問(r1,c1)爲左上角,(r2,c2)爲右下角的子矩形的最大值,並且判斷該最大值是否出現在了這個子矩陣的4個頂角上?
分析:
基本的二維RMQ應用.
注意題目中存的矩陣是無符號整數的矩陣.
<span style="font-size:18px;">#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int MAXN =305;
unsigned int val[MAXN][MAXN];
unsigned int dmax[MAXN][MAXN][9][9];
void initRMQ(int n,int m)
{
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
dmax[i][j][0][0] = val[i][j];
for(int ii=0;(1<<ii)<=n;ii++)
for(int jj=0;(1<<jj)<=m;jj++)
if(ii+jj)
for(int i=1;i+(1<<ii)-1<=n;i++)
for(int j=1;j+(1<<jj)-1<=m;j++)
if(ii)
dmax[i][j][ii][jj]=max(dmax[i][j][ii-1][jj] , dmax[i+(1<<(ii-1))][j][ii-1][jj]);
else
dmax[i][j][ii][jj]=max(dmax[i][j][ii][jj-1] , dmax[i][j+(1<<(jj-1))][ii][jj-1]);
}
unsigned int getMax(int x1,int y1,int x2,int y2)
{
int k1=0;
while((1<<(k1+1))<=x2-x1+1)k1++;
int k2=0;
while((1<<(k2+1))<=y2-y1+1)k2++;
x2 = x2-(1<<k1)+1;
y2 = y2-(1<<k2)+1;
return max(max(dmax[x1][y1][k1][k2] ,dmax[x1][y2][k1][k2] ) , max(dmax[x2][y1][k1][k2] , dmax[x2][y2][k1][k2]));
}
int main()
{
int n,m;
while(scanf("%d%d",&n,&m)==2&&n&&m)
{
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
scanf("%ud",&val[i][j]);
initRMQ(n,m);
int q;
scanf("%d",&q);
while(q--)
{
int x1,y1,x2,y2;
scanf("%d%d%d%d",&x1,&y1,&x2,&y2);
unsigned int max_num = getMax(x1,y1,x2,y2);
bool ok=false;
if(val[x1][y1]==max_num || val[x1][y2]==max_num || val[x2][y1]==max_num ||val[x2][y2]==max_num)
ok=true;
printf("%d %s\n",max_num,ok?"yes":"no");
}
}
return 0;
}
</span>