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

原文链接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;
}

  

原文地址:https://www.cnblogs.com/zhouzhendong/p/8672131.html

时间: 2024-11-07 09:32:11

BZOJ3262/洛谷P3810 陌上花开 CDQ分治 三维偏序 树状数组的相关文章

BZOJ 3262: 陌上花开 [CDQ分治 三维偏序]

Description 有n朵花,每朵花有三个属性:花形(s).颜色(c).气味(m),又三个整数表示.现要对每朵花评级,一朵花的级别是它拥有的美丽能超过的花的数量.定义一朵花A比另一朵花B要美丽,当且仅当Sa>=Sb,Ca>=Cb,Ma>=Mb.显然,两朵花可能有同样的属性.需要统计出评出每个等级的花的数量. Input 第一行为N,K (1 <= N <= 100,000, 1 <= K <= 200,000 ), 分别表示花的数量和最大属性值. 以下N行,每

BZOJ 2244 SDOI2011 拦截导弹 CDQ分治/二维树状数组

题目大意:给定一个序列,每个元素是一个二元组,等概率选择一LIS,求LIS长度以及每个元素被选中的概率 第一问CDQ分治裸上 第二问用每个元素所在的LIS个数/总LIS个数就是答案 每个元素所在的LIS自己必选,然后统计前面的方案数和后面的方案数 以前面的方案数为例,令f[x]为以x结尾的LIS长度,那么有DP方程: g[i]=Σg[j] (f[j]+1=f[i],j<i,a[j].x<a[i].x,a[j].y<a[i].y) 将所有元素按f值排序,分层DP,每层DP是一个三维偏序,上

洛谷 P3810 【模板】三维偏序(陌上花开) (cdq分治模板)

在solve(L,R)中,需要先分治solve两个子区间,再计算左边区间修改对右边区间询问的贡献. 注意,计算额外的贡献时,两子区间各自内部的顺序变得不再重要(不管怎么样左边区间的都发生在右边之前),于是就少了一维 https://www.lydsy.com/JudgeOnline/problem.php?id=3262 https://www.luogu.org/problemnew/show/P3810 此题每个操作既是修改又是查询 对于此题,先按一维排序,在solve(L,R)中先solv

洛谷 P1908 逆序对 Label:归并排序||树状数组

题目描述 猫猫TOM和小老鼠JERRY最近又较量上了,但是毕竟都是成年人,他们已经不喜欢再玩那种你追我赶的游戏,现在他们喜欢玩统计.最近,TOM老猫查阅到一个人类称之为“逆序对”的东西,这东西是这样定义的:对于给定的一段正整数序列,逆序对就是序列中ai>aj且i<j的有序对.知道这概念后,他们就比赛谁先算出给定的一段正整数序列中逆序对的数目. 输入输出格式 输入格式: 第一行,一个数n,表示序列中有n个数. 第二行n个数,表示给定的序列. 输出格式: 给定序列中逆序对的数目. 输入输出样例 输

【算法】CDQ分治 -- 三维偏序 &amp; 动态逆序对

初次接触CDQ分治,感觉真的挺厉害的. 整体思路即分而治之,再用之前处理出来的答案统计之后的答案. 大概流程是: 对于区间 l ~ r : 1.处理 l ~mid, mid + 1 ~ r 的答案 2.分别排序规整 3.计算 l ~ mid 中每一个数对 mid + 1 ~ r 中的答案的贡献, 累加 4.得到区间l ~ r的答案 CDQ分治我一共也才做了两道题目, 就一起整理在这里了.大体都差不多,CDQ+树状数组分别维护两个维度. 1.三维偏序 #include <bits/stdc++.h

CDQ分治 三维偏序

这应该是一道CDQ分治的入门题目 我们知道,二维度的偏序问题直接通过,树状数组就可以实现了,但是三维如何实现呢? 我记得以前了解过一个小故事,应该就是分治的. 一个皇帝,想给部下分配任务,但是部下太多,他也无从下手于是他这个任务分给宰相,宰相也不怎么清楚,于是他又分给他的手下,这么一直分啊分啊,分到每一个人头顶上的时候 每个人知道自己要干什么,于是他把它的信息交给他的上级,上级有了这些数据后,他处理了交给他的上级...这么一直交啊...国王最后成功的分配这些任务. CDQ分治也是一样,在这里,首

BZOJ.1935.[SHOI2007]Tree园丁的烦恼(CDQ分治 三维偏序)

题目链接 矩形查询可以拆成四个点的前缀和查询(树套树显然 但是空间不够) 每个操作表示为(t,x,y),t默认有序,对x分治,y用树状数组维护 初始赋值需要靠修改操作实现. //119964kb 4380ms #include <cstdio> #include <cctype> #include <algorithm> #define gc() getchar() #define lb(x) (x)&-(x) const int N=5e5+5; int n,

UOJ276 [清华集训2016] 汽水 【二分答案】【点分治】【树状数组】

题目分析: 这种乱七八糟的题目一看就是点分治,答案有单调性,所以还可以二分答案. 我们每次二分的时候考虑答案会不会大于等于某个值,注意到系数$k$是无意义的,因为我们可以通过转化使得$k=0$. 合并的过程相当于很多个向量,加起来后看斜率. 注意单个向量也要判定. 由于有了二分的答案$Ans$.判定变得简单多了,推一下. $-k \leq \frac{A+C}{B+D} \leq k \Rightarrow -k(B+D) \leq A+C \leq k(B+D)$. 进一步的$A+kB \ge

UVA 11990 `Dynamic&#39;&#39; Inversion CDQ分治, 归并排序, 树状数组, 尺取法, 三偏序统计 难度: 2

题目 https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem&problem=3141 题意 一个1到n的排列,每次随机删除一个,问删除前的逆序数 思路 综合考虑,对每个数点,令value为值,pos为位置,time为出现时间(总时间-消失时间),明显是统计value1 > value2, pos1 < pos2, time1 < time2的个