BZOJ3262/洛谷P3810 陌上花开 分治 三维偏序 树状数组

原文链接http://www.cnblogs.com/zhouzhendong/p/8672131.html

题目传送门 - BZOJ3262

题目传送门 - 洛谷P3810

题意

  有$n$个元素,第$i$个元素有$a_i$、$b_i$、$c_i$三个属性,设$f(i)$表示满足$a_j\leq a_i$且$b_j\leq b_i$且$c_j\leq c_i$的$j$的数量。对于$d\in [0,n)$,求$f(i)=d$的数量。

  $n\leq 100000,max\{a_i,b_i,c_i|i\in[1,n]\}<=200000$

题解

  三维偏序模版题。

  CDQ分治一波。树套树也可以。

  CDQ分治思路:

    一维偏序:直接排序。

    二维偏序:树状数组。

    三维偏序:第3维套CDQ分治。

  对于第一维和第二维我们还是按照原样处理。

  对于第三维,我们考虑分治。

  首先显然要排序。

  对于一个区间$[L,R]$,首先求得$mid=\left\lfloor\frac{L+R}{2}\right\rfloor$。

  然后考虑先分治$[L,mid]$和$[mid+1,R]$。

  然后通过树状数组的做法,用左区间来更新右区间。注意树状数组的还原。

  总体来说,对于第3维,是在开始的时候保存了rank。对于第2维,是在递归的过程中归并得到了有序排列。对于第1维,由于一开始排序的时候作为了第3关键字,而归并排序具有稳定性,所以,对于第二维相同的,第三维是有序的。

代码

#include <bits/stdc++.h>
using namespace std;
const int N=100005;
struct Node{
	int x,y,z,rank,res;
	void get(){
		scanf("%d%d%d",&x,&y,&z),res=0;
	}
}a[N],b[N];
int n,k,tree[N*2],tot[N];
bool cmp(Node a,Node b){
	if (a.x!=b.x)
		return a.x<b.x;
	if (a.y!=b.y)
		return a.y<b.y;
	return a.z<b.z;
}
bool issame(Node a,Node b){
	return a.x==b.x&&a.y==b.y&&a.z==b.z;
}
void sameadd(){
	int tot=1,last=n;
	for (int i=n-1;i>=1;i--)
		if (issame(a[i],a[last]))
			a[i].res+=tot++;
		else
			last=i,tot=1;
}
int lowbit(int x){
	return x&-x;
}
void add(int x,int y){
	for (;x<=k;x+=lowbit(x))
		tree[x]+=y;
}
int sum(int x){
	int ans=0;
	for (;x>0;x-=lowbit(x))
		ans+=tree[x];
	return ans;
}
void CDQ(int L,int R){
	if (L==R)
		return;
	int mid=(L+R)>>1,cnt=L;
	CDQ(L,mid),CDQ(mid+1,R);
	for (int i=L,l=L,r=mid+1;i<=R;i++)
		if (r>R||(l<=mid&&a[l].y<=a[r].y))
			b[cnt++]=a[l++];
		else
			b[cnt++]=a[r++];
	for (int i=L;i<=R;i++){
		a[i]=b[i];
		if (a[i].rank<=mid)
			add(a[i].z,1);
		else
			a[i].res+=sum(a[i].z);
	}
	for (int i=L;i<=R;i++)
		if (a[i].rank<=mid)
			add(a[i].z,-1);
}
int main(){
	scanf("%d%d",&n,&k);
	for (int i=1;i<=n;i++)
		a[i].get();
	sort(a+1,a+n+1,cmp);
	sameadd();
	for (int i=1;i<=n;i++)
		a[i].rank=i;
	memset(tree,0,sizeof tree);
	CDQ(1,n);
	memset(tot,0,sizeof tot);
	for (int i=1;i<=n;i++)
		tot[a[i].res]++;
	for (int i=0;i<n;i++)
		printf("%d\n",tot[i]);
	return 0;
}

  

posted @ 2018-03-29 20:13  zzd233  阅读(288)  评论(0编辑  收藏  举报