F Planting Trees

题目链接

https://ac.nowcoder.com/acm/contest/883/F

题目描述

The semester is finally over and the summer holiday is coming. However, as part of your university's graduation requirement, you have to take part in some social service during the holiday. Eventually, you decided to join a volunteer group which will plant trees in a mountain.

To simplify the problem, let's represent the mountain where trees are to be planted with an \(N \times N\) grid. Let's number the rows \(\ 1\) to \(\ N\) from top to bottom, and number the columns \(\ 1\) to \(\ N\) from left to right. The elevation of the cell in the \(\ i\)-th row and \(\ j\)-th column is denoted by \(a_{i,j}\). Your leader decides that trees should be planted in a rectangular area within the mountain and that the maximum difference in elevation among the cells in that rectangle should not exceed M. In other words, if the coordinates of the top-left and the bottom-right corners of the rectangle are \((x_1,y_1)\) and \((x_2,y_2)\), then the condition $ |a_{i,j} - a_{k,l}| \le M$ must hold for \(x_1 \le i,k \le x_2, \ y_1 \le j,l \le y_2\). Please help your leader calculate the maximum possible number of cells in such a rectangle so that he'll know how many trees will be planted.

输入描述:

The input contains multiple cases. The first line of the input contains a single integer \(T\ (1 \le T \le 1000)\), the number of cases.

For each case, the first line of the input contains two integers $ N\ (1 \le N \le 500)$ and \(M\ (0 \le M \le 10^5)\). The following N lines each contain N integers, where the\(\ j-th\) integer in the i\ i i-th line denotes \(a_{i,j} \ (1 \le a_{i,j} \le 10^5)\).

It is guaranteed that the sum of \(N^3\) over all cases does not exceed \(25 \cdot 10^7\).

输出描述:

For each case, print a single integer, the maximum number of cells in a valid rectangle.

输入

    2
2 0
1 2
2 1
3 1
1 3 2
2 3 1
3 2 1

输出

    1
4

题意

给你一个\(N \times N\)的矩阵,求最大的自矩阵满足子矩阵中最大值和最小值只差小于m.

题解

一看体面最后一句话,寻思着就是\(n^3\)的算法,于是很自然地想到\(n^2\)枚举上届和下界,再用\(n\)来扫描一遍。

我们想想假如上下界确定了,我们再枚举左边界,右边界越长越好,那么左边界往右移时,右边界一定不会往左移(这里应该不难想,只要想肯定不难想到),这个好像就叫做尺取法。

剩下的就是要求区间的最小值和最大值了,我一开始直接二维倍增,结果超时了,改了两个小时还是超时,只能放弃了。之后用了单调队列来求最小值和最大值,这种做法感觉很经典,但是我竟然没看出来,还是做少了题目.

简要地讲讲单调队列:

我们搞两个队列,一个用来求最大值,另一个用来求最小值

第一步:去掉不合法的队首(更改左边界)

第二部:往后延伸右边界

第三部:更新答案

这么讲有点抽象,之后我去找找基础的题目,做完基础题目,这个自然也就会了。

AC代码

#include<bits/stdc++.h>
using namespace std;
#define ll long long
#define INF 0x7f7f7f7f
#define N 505
#define maxx(a,b) a<b?b:a
#define minn(a,b) a<b?a:b
int n,m,mx[N],mi[N],a[N][N];
struct Queue{int val,id;}Q1[N],Q2[N];
clock_t now;
template<typename T>void read(T&x)
{
ll k=0; char c=getchar();
x=0;
while(!isdigit(c)&&c!=EOF)k^=c=='-',c=getchar();
if (c==EOF)exit(0);
while(isdigit(c))x=x*10+c-'0',c=getchar();
x=k?-x:x;
}
inline void work()
{
now=clock();
int ans=1;
read(n); read(m);
for(int i=1;i<=n;i++)
for(int j=1;j<=n;j++)
read(a[i][j]);
for(int i=1;i<=n;i++)
for(int j=i;j<=n;j++)
{
int ds1=1,dw1=0,ds2=1,dw2=0,r=0;
for(int k=1;k<=n;k++)
{
if (i==j)mx[k]=mi[k]=a[i][k];
else {mx[k]=maxx(mx[k],a[j][k]);mi[k]=minn(mi[k],a[j][k]);}
}
for(int k=1;k<=n;k++)
{
r=maxx(r,k-1);
while(ds1<=dw1&&Q1[ds1].id<k)ds1++;
while(ds2<=dw2&&Q2[ds2].id<k)ds2++;
if (mx[k]-mi[k]>m)continue;
int m1=maxx(mx[r+1],Q1[ds1].val);
int m2=minn(mi[r+1],Q2[ds2].val);
if (ds1>dw1)m1=mx[r+1],m2=mi[r+1];
while(r+1<=n&&(mx[r+1]-mi[r+1]<=m)&&(ds1>dw1||m1-m2<=m))
{
while(mx[r+1]>Q1[dw1].val&&ds1<=dw1)dw1--;
while(mi[r+1]<Q2[dw2].val&&ds2<=dw2)dw2--;
Q1[++dw1]={mx[r+1],r+1};
Q2[++dw2]={mi[r+1],r+1};
r++;
m1=maxx(m1,mx[r+1]);
m2=minn(m2,mi[r+1]);
}
ans=maxx(ans,(j-i+1)*(r-k+1));
} }
printf("%d\n",ans);
//cout<<(double)(clock()-now)/CLOCKS_PER_SEC*1000<<endl;
}
int main()
{
#ifndef ONLINE_JUDGE
freopen("aa.in","r",stdin);
#endif
int T;
read(T);
while(T--)work();
}

超时代码(仅供观赏)

#include<bits/stdc++.h>
using namespace std;
#define ll long long
#define INF 0x7f7f7f7f
#define N 505
int n,m,mx[N][N][10][10],mi[N][N][10][10],mm[N],mv[10];
template<typename T>void read(T&x)
{
ll k=0; char c=getchar();
x=0;
while(!isdigit(c)&&c!=EOF)k^=c=='-',c=getchar();
if (c==EOF)exit(0);
while(isdigit(c))x=x*10+c-'0',c=getchar();
x=k?-x:x;
}
void read_char(char &c)
{while(!isalpha(c=getchar())&&c!=EOF);}
void build_ST()
{
for(int i=2;i<=n;i++)mm[i]=mm[i>>1]+1;
for(int i=0;i<=10;i++)mv[i]=1<<i;
for(int i=1;i<=n;i++)
for(int j=1;j<=n;j++)
mx[i][j][0][0]=mi[i][j][0][0];
for(int k=1;k<=8;k++)
for(int i=1;i<=n;i++)
for(int j=1;j+(1<<k)-1<=n;j++)
{
mx[i][j][0][k]=max(mx[i][j][0][k-1],mx[i][j+(1<<(k-1))][0][k-1]);
mi[i][j][0][k]=min(mi[i][j][0][k-1],mi[i][j+(1<<(k-1))][0][k-1]);
} for(int k1=1;k1<=8;k1++)
for(int k2=0;k2<=8;k2++)
for(int i=1;i+(1<<k1)-1<=n;i++)
for(int j=1;j+(1<<k2)-1<=n;j++)
{
mx[i][j][k1][k2]=
max(mx[i][j][k1-1][k2],mx[i+(1<<(k1-1))][j][k1-1][k2]);
mi[i][j][k1][k2]=
min(mi[i][j][k1-1][k2],mi[i+(1<<(k1-1))][j][k1-1][k2]);
} }
inline int get_mx(int x1,int y1,int x2,int y2)
{
int k1=mm[x2-x1+1],k2=mm[y2-y1+1];
x2=x2-mv[k1]+1;
y2=y2-mv[k2]+1;
int a1=max(mx[x1][y1][k1][k2],mx[x1][y2][k1][k2]);
int a2=max(mx[x2][y1][k1][k2],mx[x2][y2][k1][k2]);
return max(a1,a2);
}
inline int get_mi(int x1,int y1,int x2,int y2)
{
int k1=mm[x2-x1+1],k2=mm[y2-y1+1];
x2=x2-(1<<k1)+1;
y2=y2-(1<<k2)+1;
int a1=min(mi[x1][y1][k1][k2],mi[x1][y2][k1][k2]);
int a2=min(mi[x2][y1][k1][k2],mi[x2][y2][k1][k2]);
return min(a1,a2);
}
void work()
{
int ans=1;
read(n); read(m);
for(int i=1;i<=n;i++)
for(int j=1;j<=n;j++)
read(mi[i][j][0][0]);
build_ST();
for(int i=1;i<=n;i++)
for(int j=i;j<=n;j++)
{
int l=1;
for(int k=1;k<=n;k++)
{
int mx=get_mx(i,l,j,k);
int mi=get_mi(i,l,j,k);
while(mx-mi>m&&l<=k)
{
l++;
if(l<=k)mx=get_mx(i,l,j,k);
if(l<=k)mi=get_mi(i,l,j,k);
}
ans=max(ans,(j-i+1)*(k-l+1));
}
}
printf("%d\n",ans);
}
int main()
{
#ifndef ONLINE_JUDGE
freopen("aa.in","r",stdin);
#endif
int T;
read(T);
while(T--)work();
}

2019牛客暑期多校训练营(第三场)F Planting Trees 单调队列的更多相关文章

  1. 2019牛客暑期多校训练营(第三场) F.Planting Trees(单调队列)

    题意:给你一个n*n的高度矩阵 要你找到里面最大的矩阵且最大的高度差不能超过m 思路:我们首先枚举上下右边界,然后我们可以用单调队列维护一个最左的边界 然后计算最大值 时间复杂度为O(n*n*n) # ...

  2. 2019牛客暑期多校训练营(第三场)- F Planting Trees

    题目链接:https://ac.nowcoder.com/acm/contest/883/F 题意:给定n×n的矩阵,求最大子矩阵使得子矩阵中最大值和最小值的差值<=M. 思路:先看数据大小,注 ...

  3. 2019牛客暑期多校训练营(第一场)-A (单调栈)

    题目链接:https://ac.nowcoder.com/acm/contest/881/A 题意:给定两个长度均为n的数组a和b,求最大的p使得(a1,ap)和(b1,bp)等价,等价的定义为其任意 ...

  4. 2019牛客暑期多校训练营(第三场)H题目

    题意:给你一个N×N的矩阵,求最大的子矩阵 满足子矩阵中最大值和最小值之差小于等于m. 思路:这题是求满足条件的最大子矩阵,毫无疑问要遍历所有矩阵,并判断矩阵是某满足这个条件,那么我们大致只要解决两个 ...

  5. 2019牛客暑期多校训练营(第九场)A:Power of Fibonacci(斐波拉契幂次和)

    题意:求Σfi^m%p. zoj上p是1e9+7,牛客是1e9:  对于这两个,分别有不同的做法. 前者利用公式,公式里面有sqrt(5),我们只需要二次剩余求即可.     后者mod=1e9,5才 ...

  6. 2019牛客暑期多校训练营(第一场)A题【单调栈】(补题)

    链接:https://ac.nowcoder.com/acm/contest/881/A来源:牛客网 题目描述 Two arrays u and v each with m distinct elem ...

  7. 2019牛客暑期多校训练营(第一场) B Integration (数学)

    链接:https://ac.nowcoder.com/acm/contest/881/B 来源:牛客网 Integration 时间限制:C/C++ 2秒,其他语言4秒 空间限制:C/C++ 5242 ...

  8. 2019牛客暑期多校训练营(第一场) A Equivalent Prefixes ( st 表 + 二分+分治)

    链接:https://ac.nowcoder.com/acm/contest/881/A 来源:牛客网 Equivalent Prefixes 时间限制:C/C++ 2秒,其他语言4秒 空间限制:C/ ...

  9. 2019牛客暑期多校训练营(第二场)F.Partition problem

    链接:https://ac.nowcoder.com/acm/contest/882/F来源:牛客网 Given 2N people, you need to assign each of them ...

  10. 2019牛客暑期多校训练营(第一场)A Equivalent Prefixes(单调栈/二分+分治)

    链接:https://ac.nowcoder.com/acm/contest/881/A来源:牛客网 Two arrays u and v each with m distinct elements ...

随机推荐

  1. 线程的分离状态 detached joinable

    转自  http://blog.chinaunix.net/uid-26983585-id-3315953.html 其实在写上一篇日志的时候,由于我把创建线程的返回值的判断条件写错了,程序每次运行的 ...

  2. cdh版hbase构建Phoenix 遇到的坑

    Phoenix 构建cdh版hbase遇到的坑 1. 安装phoenix 下载:在github上下载对应版本https://github.com/apache/phoenix 解压:略 编译: 修改根 ...

  3. ReactJS和AngularJS对比

    Angular的特点: 优势: AngularJS是一套完整的框架,angular有自带的数据绑定.render渲染.angularUI库,过滤器,$filter,$directive(模板),$se ...

  4. Mac 卸载Python3.6

    Mac 自带的 Python 已经能够满足我们的需要了,因此很多同学在安装完 Python 之后,又想要将其删除,或者称之为卸载. 对于删除 Python,我们首先要知道其具体都安装了什么,实际上,在 ...

  5. /proc/sys/net/ipv4/ip_conntrack_max

    Things to know (best practices and “issues”) READ IT !!! — uWSGI 2.0 documentationhttps://uwsgi-docs ...

  6. js中string和json的相互转换

    1.将string转成json var json={}; var myorderno=$("#myorderno").val(); json.myorderno=myorderno ...

  7. Jetson TK下如何写汇编语言

    首先,可以根据http://www.cnblogs.com/zenny-chen/p/3816620.html来安装CUDA工具链.这个工具集里包含了CUDA编译器以及其它必要的工具.然后,我们进入/ ...

  8. [spring]数据库的连接配置

    使用druid数据源 ,并支持事务处理. <?xml version="1.0" encoding="UTF-8"?> <beans xmln ...

  9. Idea创建SpringBoot项目整合Hibernate

    然后next 依赖包选择,Web必须 pom.xml: <?xml version="1.0" encoding="UTF-8"?> <pro ...

  10. 关于Python正则表达式findall函数问题详解

    关于Python正则表达式 findall函数问题详解 在写正则表达式的时候总会遇到不少的问题, 特别是在表达式有多个元组的时候.下面看下re模块下的findall()函数和多个表达式元组相遇的时候会 ...