【BZOJ4237】稻草人 cdq分治+单调栈+二分

【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

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;
}
原文地址:https://www.cnblogs.com/CQzhangyu/p/7123515.html