【BZOJ4237】稻草人
Description
JOI村有一片荒地,上面竖着N个稻草人,村民们每年多次在稻草人们的周围举行祭典。
有一次,JOI村的村长听到了稻草人们的启示,计划在荒地中开垦一片田地。和启示中的一样,田地需要满足以下条件:
田地的形状是边平行于坐标轴的长方形;
左下角和右上角各有一个稻草人;
田地的内部(不包括边界)没有稻草人。
给出每个稻草人的坐标,请你求出有多少遵从启示的田地的个数
Input
第一行一个正整数N,代表稻草人的个数
接下来N行,第i行(1<=i<=N)包含2个由空格分隔的整数Xi和Yi,表示第i个稻草人的坐标
Output
输出一行一个正整数,代表遵从启示的田地的个数
Sample Input
4
0 0
2 2
3 4
4 3
0 0
2 2
3 4
4 3
Sample Output
3
HINT
所有满足要求的田地由下图所示:
1<=N<=2*10^5
0<=Xi<=10^9(1<=i<=N)
0<=Yi<=10^9(1<=i<=N)
Xi(1<=i<=N)互不相同。
Yi(1<=i<=N)互不相同。
题解:这种题能否想到cdq分治是重点。
既然用cdq分治就一定要排序,那么怎么排呢?我们发现:第一维按x从小到大排序,第二位按y从大到小排序,可以便于我们进一步的处理。(反正就4种排序方法,自己挨个试一试就知道了~)
所以我们将所有点按x分治,左右两边分别按y从大到小排序,我们想找的就是左下角在[l,mid],右上角在[mid+1,r]中的矩形。发现,如果[l,mid]中的一个点i可以跟[mid+1,r]中的一些j1,j2...jk组成矩形,那么假设y[j1]>y[j2]>...>y[jk],一定满足x[j1]<x[j2]<..<x[jk]。反过来,每个j对应的i1,i2...ik也是有规律的。所以我们想到给左右两边都维护一个单调栈,那么思考该维护两个怎样的单调栈。
经过尝试发现(也就4种,一个一个试~),令左边的单调栈中的x单调递减,右边的x单调递增,可以便于我们进一步处理。我们对于[l,mid]中的i,设k是[l,mid]中满足x[k]>x[i],y[k]>y[i]且y值最小的点。那么i只能与[mid+1,r]中y值在[y[i],y[k]]中的点组成矩形。并且,如果这些点的y值递减,这些点的x值是递增的(也就是说都在单调栈里)。那么我们直接在右边的单调栈中二分统计一下y值符合条件的个数就行了。
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;
const int maxn=200010;
typedef long long ll;
ll ans;
int n,t1,t2;
int x[maxn],y[maxn],p[maxn],s1[maxn],s2[maxn];
int rd()
{
int ret=0,f=1; char gc=getchar();
while(gc<'0'||gc>'9') {if(gc=='-')f=-f; gc=getchar();}
while(gc>='0'&&gc<='9') ret=ret*10+gc-'0',gc=getchar();
return ret*f;
}
bool cmpx(int a,int b)
{
return (x[a]==x[b])?(y[a]>y[b]):(x[a]<x[b]);
}
bool cmpy(int a,int b)
{
return (y[a]==y[b])?(x[a]<x[b]):(y[a]>y[b]);
}
void solve(int l,int r)
{
if(l==r) return ;
int i,j,mid=l+r>>1,last,L,R,MID;
sort(p+l,p+mid+1,cmpy),sort(p+mid+1,p+r+1,cmpy);
for(i=l,j=mid+1,t1=t2=0;i<=mid;i++)
{
for(;y[p[j]]>=y[p[i]]&&j<=r;j++)
{
while(t2&&x[s2[t2]]>x[p[j]]) t2--;
s2[++t2]=p[j];
}
while(t1&&x[s1[t1]]<x[p[i]]) t1--;
last=y[s1[t1]],s1[++t1]=p[i];
L=1,R=t2+1;
while(L<R)
{
MID=L+R>>1;
if(y[s2[MID]]<=last) R=MID;
else L=MID+1;
}
ans+=t2-R+1;
}
sort(p+l,p+r+1,cmpx);
solve(l,mid),solve(mid+1,r);
}
int main()
{
x[0]=0,y[0]=1<<30;
n=rd();
int i;
for(i=1;i<=n;i++) x[i]=rd(),y[i]=rd(),p[i]=i;
sort(p+1,p+n+1,cmpx);
solve(1,n);
printf("%lld",ans);
return 0;
}