学习笔记——CDQ分治
再次感谢这位大佬的博客:https://www.cnblogs.com/ljc20020730/p/10395866.html
CDQ分治,是一种在分治合并中计算前面值对后面答案的贡献的一种算法。今天主要围绕多维偏序问题来对CDQ分治进行介绍
先定义偏序:(以下转载自百度百科)
设R是集合A上的一个关系,如果R是自反的、反对称的和可传递的,则称R是集合A的偏序关系,简称偏序
二维偏序:给定n个二元组,求有多少对二元组满足$a[i].x>=a[j].x$,且$a[i].y>=a[j].y$
暴力$n^{2}$肯定不行,我们可以采用归并排序的方法,对第一维从小到大进行排序,这样只会前面影响后面,然后我们再用类似于“逆序对”的方法统计第二就可以啦~~
拓展题:CF957E(怎么用二维偏序自己想哦~)
三位偏序:和二维偏序类似,在用逆序对维护第二维的同时,我们需要再用数据结构统计一下同时满足第三维时的贡献
做法:
1.对第一维进行排序
2.对二维进行归并排序,并同时统计前面对后面的贡献(在同一组的之前已经统计过了):
3.若前面$a[i].y$一直小于等于$a[j].y$,则将$a[i].z$塞到树状数组里维护:即这个这段区间内值为$z$的有几个(记得对于$z$离散化)
4.一旦$a[i]>a[j]$就说明后面的$a[i]$不会再对$a[j]$产生贡献,所以对于$a[j].z$在树状数组$[1,a[j].z]$区间内查询有多少个$z$即为贡献
代码实现:(细节超多啊QAQ)
首先是要去重,因为完全相同的三元组是会互相影响的,而在归并排序中只会左边影响右边,会少讨论情况
其次是在统计答案时每几个完全相同的三元组的内部影响要加上
还有就是在树状数组用完以后要清0,但不要用memset,不然复杂度会直线上升,只需要对于之前更新过的树状数组再反着更新回来就可以啦~
模板题:陌上花开
#include<bits/stdc++.h>
using namespace std;
const int N=;
void print()
{
puts("OK");
} int n,tt[N],ans[N],f[N],k;
int C[N*];
struct node{
int x,y,z,id,cnt;
bool operator < (const node &rhs) const{
if(x!=rhs.x) return x<rhs.x;
if(y!=rhs.y) return y<rhs.y;
if(z!=rhs.z) return z<rhs.z;
return id<rhs.id;
}
bool operator == (const node &rhs) const{
return x==rhs.x&&y==rhs.y&&z==rhs.z;
}
}a[N];
node t[N]; int tot=; int lowbit(int x)
{
return x&(-x);
} void update(int p,int x)
{
while(p<=k)//树状数组长度为k(值不同的个数)
{
C[p]+=x;
p+=lowbit(p);
}
} int query(int x)
{
int ret=;
while(x>)
{
ret+=C[x];
x-=lowbit(x);
}
return ret;
} void init()
{
sort(a+,a+n+);
int i=; tot=;
while(i<=n)
{
int j=i+;
while(j<=n&&a[j]==a[i]) j++;
t[++tot]=a[i];
t[tot].cnt=j-i;
t[tot].id=tot;
i=j;
}
for(int i=;i<=tot;i++) a[i]=t[i];
for(int i=;i<=tot;i++) tt[i]=a[i].z;
sort(tt+,tt+tot+);
int m=unique(tt+,tt+tot+)-tt-;
for(int i=;i<=tot;i++)
{
a[i].z=lower_bound(tt+,tt+m+,a[i].z)-tt;
//记得要算自己内部的影响
}
} void CDQ(int l,int r)
{
if(l==r) return;
int mid=(l+r)>>;
CDQ(l,mid);
CDQ(mid+,r);
int i=l,j=mid+,top=l-;
while(i<=mid&&j<=r)
{
if(a[i].y<=a[j].y)//能做出贡献
{
update(a[i].z,a[i].cnt);
t[++top]=a[i++];
}
else
{
ans[a[j].id]+=query(a[j].z);//之前j写成i了
t[++top]=a[j++];
}
}
while(i<=mid)
{
update(a[i].z,a[i].cnt);
t[++top]=a[i];
i++;
}
while(j<=r)
{
ans[a[j].id]+=query(a[j].z);
t[++top]=a[j];
j++;
}
//for(int i=1;i<=tot;i++) cout<<ans[i]<<" "; cout<<endl;
for(int i=l;i<=mid;i++)
{
update(a[i].z,-a[i].cnt);
}
for(int i=l;i<=r;i++) a[i]=t[i];
} int main()
{
memset(ans,,sizeof(ans));
memset(f,,sizeof(f));
scanf("%d%d",&n,&k);
for(int i=;i<=n;i++)
{
scanf("%d%d%d",&a[i].x,&a[i].y,&a[i].z);
a[i].cnt=;
}
init();
CDQ(,tot);
for(int i=;i<=tot;i++)
{
ans[a[i].id]+=a[i].cnt-;//之前枚举ans的坐标,却发现没有保存原坐标对应现坐标,但是现a[i]有对应原坐标,所以枚举a下标
f[ans[a[i].id]]+=a[i].cnt;//现顺序已经被归并排序弄得毫无意义,要用id,加cnt,而不是1
}
for(int i=;i<n;i++) printf("%d\n",f[i]);
return ;
}
四维偏序:这是神仙题啊~~~一般用bitset解决,CDQ套CDQ复杂度实在是太大了,但还是介绍一下思路吧
首先第一维还是排序,然后对第二位进行CDQ分治,但是还剩下两维(可以每一次插入都把后三位拉出来进行cdq,但复杂度太高了,我们考虑在最后进行)
因为只有前面对后面有影响,所以我们记录它是在左边还是在右边
显然在归并排序之后第二维已经达到了有序,然后我们再来一遍CDQ,对记录是左边的进行update,右边的进行query,而且因为从第二维小到大排序,所以一定只有前面影响后面
最后就是再对第三位进行分治,然后对第四维进行树状数组统计就可以啦~~
(代码详见上述那位神仙的博客,本蒟蒻没有这个实力啊QAQ)
学习笔记——CDQ分治的更多相关文章
- 学习笔记 | CDQ分治
目录 前言 啥是CDQ啊(它的基本思想) 例题 后记 参考博文 前言 博主太菜了 学习快一年的OI了 好像没有什么会的算法 更寒碜的是 学一样还不精一样TAT 如有什么错误请各位路过的大佬指出啊感谢! ...
- [学习笔记] CDQ分治 从感性理解到彻底晕菜
最近学了一种叫做CDQ分治的东西...用于离线处理一系列操作与查询似乎跑得很快233 CDQ的名称似乎源于金牌选手陈丹琦 概述: 对于一坨操作和询问,分成两半,单独处理左半边和处理左半边对于右半边的影 ...
- [学习笔记]CDQ分治和整体二分
序言 \(CDQ\) 分治和整体二分都是基于分治的思想,把复杂的问题拆分成许多可以简单求的解子问题.但是这两种算法必须离线处理,不能解决一些强制在线的题目.不过如果题目允许离线的话,这两种算法能把在线 ...
- [学习笔记] CDQ分治&整体二分
突然诈尸.png 这两个东西好像都是离线骗分大法... 不过其实这两个东西并不是一样的... 虽然代码长得比较像 CDQ分治 基本思想 其实CDQ分治的基本思想挺简单的... 大概思路就是长这样的: ...
- 算法笔记--CDQ分治 && 整体二分
参考:https://www.luogu.org/blog/Owencodeisking/post-xue-xi-bi-ji-cdq-fen-zhi-hu-zheng-ti-er-fen 前置技能:树 ...
- 一篇自己都看不懂的CDQ分治&整体二分学习笔记
作为一个永不咕咕咕的博主,我来更笔记辣qaq CDQ分治 CDQ分治的思想还是比较简单的.它的基本流程是: \(1.\)将所有修改操作和查询操作按照时间顺序并在一起,形成一段序列.显然,会影响查询操作 ...
- 【教程】简易CDQ分治教程&学习笔记
前言 辣鸡蒟蒻__stdcall终于会CDQ分治啦! CDQ分治是我们处理各类问题的重要武器.它的优势在于可以顶替复杂的高级数据结构,而且常数比较小:缺点在于必须离线操作. CDQ分治的基 ...
- [偏序关系与CDQ分治]【学习笔记】
组合数学真是太棒了 $CDQ$真是太棒了(雾 参考资料: 1.<组合数学> 2.论文 课件 很容易查到 3.sro __stdcall 偏序关系 关系: 集合$X$上的关系是$X$与$X$ ...
- 初学cdq分治学习笔记(可能有第二次的学习笔记)
前言骚话 本人蒟蒻,一开始看到模板题就非常的懵逼,链接,学到后面就越来越清楚了. 吐槽,cdq,超短裙分治....(尴尬) 正片开始 思想 和普通的分治,还是分而治之,但是有一点不一样的是一般的分治在 ...
随机推荐
- 树的直径变形——cf1238F
/* 题目给定一些一维线段[li,ri],要求从这些线段里挑出一些线段,每条线段对应一个点,如果两线段相交,那么点连边,这样得到的树是good-tree 现在给定一棵树,要求从该树中选出一棵子树,使这 ...
- elementUI拿到当前表格行的数据的另一种写法
背景: 这里是通过点击“修改”按钮后才拿到当前行的数据,不是点击当前行任意位置拿到数据,所以不能用 @row-click 方法 改用点击的时候直接拿到这个表里面的这一条数据 1.绑定事件 <te ...
- iOS 几种定时器
//第一种 每一秒执行一次(重复性) double delayInSeconds = 1.0; timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_ ...
- 4、APP FPS测试
什么是FPS FPS是图像领域中的定义,是指画面每秒传输帧数,通俗来讲就是指动画或视频的画面数.FPS是测量用于保存.显示动态视频的信息数量.每秒钟帧数愈多,所显示的动作就会愈流畅.通常,要避免动作不 ...
- Spark中的多线程并发处理
Spark中的多任务处理 Spark的一个非常常见的用例是并行运行许多作业. 构建作业DAG后,Spark将这些任务分配到多个Executor上并行处理.但这并不能帮助我们在同一个Spark应用程序中 ...
- 3-vim-打开和新建文件-02-删除交换文件
vim的异常处理 如果vim异常退出,在磁盘上可能会保存有交换文件. 若使用vi编辑该文件时看到如下图信息,按下字母d就可以删除交换文件. 注意:输入命令操作的时候关闭输入法.
- 13、如何拆分含有多种分隔符的字符串 14、如何判断字符串a是否以字符串b开头或结尾 15、如何调整字符串中文本的格式 16、如何将多个小字符串拼接成一个大的字符串
13.如何拆分含有多种分隔符的字符串 import re s = "23:41:2314\1234#sdf\23;" print(re.split(r'[#:\;]+',s)) ...
- 【gcc】更新下载编译gcc遇到的各种问题
帮学长的oj升级gcc版本.遇到了贼多问题.. [悲惨的开始] 安装gcc版本推荐ustc的mirror的下载,超快der... https://mirrors.ustc.edu.cn/gnu/gcc ...
- Flyway - Version control for your database
Flyway 是什么? Flyway是个数据库版本管理工具.在开发过程中,数据库难免发生变更,例如数据变更,表结构变更.新建表或者视图等等. 在项目进行时无法保证一旦开发环境中的数据库内容变化候会去测 ...
- JAVA javah
{ 用法: javah [options] <classes>其中, [options] 包括: -o <file> 输出文件 (只能使用 - ...