CDQ解决一些三维偏序的问题
本来几天前就该记录的东西,硬生生被我拖到了现在,太懒了。。。
在cdq学习时,二维偏序已经解决了,无非就是先sort使第一维有序,然后再用cdq或者数据结构处理第二维。而三维偏序的时候呢,大佬的做法好像有树套树之类的,但是我不会,所以我选择cdq。
大体的思路很简单,首先也是先使第一维有序,然后cdq把第二维由小到大合并,这时再套个线段树或者树状数组处理第三维。来几题做一下
题目大意:一个花园里有n棵数,然后有m次询问,每次询问一个矩形里有多少颗树。
把原有的树视为更新操作,然后每个查询分为加上(0,0)到(x1-1,y1-1)范围内的树,减去(0,0)到(x1-1,y2)范围内的树,减去(0,0)到(x2,y1-1)范围内的树,加上(0,0)到(x2,y2)范围内的树,这样就可以处理一个二维前缀和,然后用cdq分治理套树状数组处理y轴
- #include<cstdio>
- #include<algorithm>
- #define lowb(x) (x&(-x))
- using namespace std;
- const int N=,M=,Y=,Q=(M<<)+N;//每个查询分4个
- struct Nop{
- int x,y,op,w,id;
- friend bool operator < (const Nop &n1,const Nop &n2){
- return n1.x==n2.x ? n1.op<n2.op : n1.x<n2.x;
- }
- }P[Q],temp[Q];
- int pn,maxy,ans[M<<],sumy[Y]={};
- void addp(int x,int y,int op,int w,int id){
- P[pn++]=(Nop){x,y,op,w,id};
- }
- void updata(int y)
- {
- while(y<=maxy)
- {
- sumy[y]++;
- y+=lowb(y);
- }
- }
- int getsum(int y)
- {
- int sum=;
- while(y)
- {
- sum+=sumy[y];
- y-=lowb(y);
- }
- return sum;
- }
- void clear(int y)
- {
- while(y<=maxy)
- {
- if(sumy[y])
- sumy[y]=;
- else
- break;
- y+=lowb(y);
- }
- }
- void cdq(int l,int r)
- {
- if(l==r)
- return ;
- int m=(l+r)>>;
- cdq(l,m);
- cdq(m+,r);
- int i=l,j=m+,k=l;
- while(i<=m&&j<=r)
- {
- if(P[i]<P[j])
- {
- if(P[i].op==)
- updata(P[i].y);
- temp[k++]=P[i++];
- }
- else
- {
- if(P[j].op==)
- ans[P[j].id]+=P[j].w*getsum(P[j].y);
- temp[k++]=P[j++];
- }
- }
- while(i<=m)
- temp[k++]=P[i++];
- while(j<=r)
- {
- if(P[j].op==)
- ans[P[j].id]+=P[j].w*getsum(P[j].y);
- temp[k++]=P[j++];
- }
- for(i=l;i<=r;i++)
- {
- clear(P[i].y);
- P[i]=temp[i];
- }
- }
- int main()
- {
- int n,m,x1,y1,x2,y2;
- while(~scanf("%d%d",&n,&m))
- {
- pn=maxy=;
- for(int i=;i<n;i++)
- {
- scanf("%d%d",&x1,&y1);
- x1++,y1++;
- addp(x1,y1,,,);
- maxy=max(maxy,y1);
- }
- for(int i=;i<m;i++)
- {
- ans[i]=;
- scanf("%d%d%d%d",&x1,&y1,&x2,&y2);
- x1++,y1++,x2++,y2++;
- //拆分成四个查询
- addp(x1-,y1-,,,i);
- addp(x1-,y2,,-,i);
- addp(x2,y1-,,-,i);
- addp(x2,y2,,,i);
- maxy=max(maxy,max(y1,y2));
- }
- cdq(,pn-);
- for(int i=;i<m;i++)
- printf("%d\n",ans[i]);
- }
- return ;
- }
园丁你为何烦恼呢
当然,这题很明显是个二维偏序问题,因为树本身都存在了,没有什么更新操作,直接离线树状数组处理就行。
- #include<cstdio>
- #include<algorithm>
- #define lowb(x) (x&(-x))
- using namespace std;
- const int N=,M=,Y=,Q=(M<<)+N;
- struct Nop{
- int x,y,op,w,id;
- friend bool operator < (const Nop &n1,const Nop &n2){
- return n1.x==n2.x ? n1.op<n2.op : n1.x<n2.x;
- }
- }P[Q];
- int pn,maxy,ans[M<<],sumy[Y]={};
- void addp(int x,int y,int op,int w,int id){
- P[pn++]=(Nop){x,y,op,w,id};
- }
- void updata(int y)
- {
- while(y<=maxy)
- {
- sumy[y]++;
- y+=lowb(y);
- }
- }
- int getsum(int y)
- {
- int sum=;
- while(y)
- {
- sum+=sumy[y];
- y-=lowb(y);
- }
- return sum;
- }
- void clear(int y)
- {
- while(y<=maxy)
- {
- if(sumy[y])
- sumy[y]=;
- else
- break;
- y+=lowb(y);
- }
- }
- int main()
- {
- int n,m,x1,y1,x2,y2;
- while(~scanf("%d%d",&n,&m))
- {
- pn=maxy=;
- for(int i=;i<n;i++)
- {
- scanf("%d%d",&x1,&y1);
- x1++,y1++;
- addp(x1,y1,,,);
- maxy=max(maxy,y1);
- }
- for(int i=;i<m;i++)
- {
- ans[i]=;
- scanf("%d%d%d%d",&x1,&y1,&x2,&y2);
- x1++,y1++,x2++,y2++;
- addp(x1-,y1-,,,i);
- addp(x1-,y2,,-,i);
- addp(x2,y1-,,-,i);
- addp(x2,y2,,,i);
- maxy=max(maxy,max(y1,y2));
- }
- for(int i=;i<=maxy;i++)
- clear(i);
- sort(P,P+pn);
- for(int i=;i<pn;i++)
- {
- if(!P[i].op)
- updata(P[i].y);
- else
- ans[P[i].id]+=P[i].w*getsum(P[i].y);
- }
- for(int i=;i<m;i++)
- printf("%d\n",ans[i]);
- }
- return ;
- }
园丁不烦恼了
题目大意:维护一个W*W的矩阵,初始值均为S.每次操作可以增加某格子的权值,或询问某子矩阵的总权值.修改操作数M<=160000,询问数Q<=10000,W<=2000000.
S值并没有实际作用忽略就好。这题就不行像上面那题单纯的离线然后树状数组维护就行了,因为是有着一个更新的存在,不同的查询前面的更新是不一样的。所以就是我们大体的思路,把每个操作视为(操作时间,x轴,y轴)这样的带有附加消息的三维有序对,这样第一维已经有序了,我们就只需要把x轴cdq分治处理,然后y轴有树状数组处理,查询也就是维护个二维的前缀和。
- #include<cstdio>
- #include<algorithm>
- #define lowb(x) (x&(-x))
- using namespace std;
- const int N=,M=,Y=,Q=(M<<)+N;
- struct Nop{
- int x,y,op,w,id;
- friend bool operator < (const Nop &n1,const Nop &n2){
- return n1.x==n2.x ? n1.op<n2.op : n1.x<n2.x;
- }
- }P[Q],temp[Q];
- int pn,qn,maxy,ans[M]={},sum[Y]={};
- void addp(int x,int y,int op,int w,int id){
- P[pn++]=(Nop){x,y,op,w,id};
- }
- void updata(int y,int val)
- {
- while(y<=maxy)
- {
- sum[y]+=val;
- y+=lowb(y);
- }
- }
- int getsum(int y)
- {
- int ans=;
- while(y)
- {
- ans+=sum[y];
- y-=lowb(y);
- }
- return ans;
- }
- void clear(int y)
- {
- while(y<=maxy)
- {
- if(sum[y])
- sum[y]=;
- else
- break;
- y+=lowb(y);
- }
- }
- void cdq(int l,int r)
- {
- if(l==r)
- return ;
- int m=(l+r)>>;
- cdq(l,m);
- cdq(m+,r);
- int i=l,j=m+,k=l;
- while(i<=m&&j<=r)
- {
- if(P[i]<P[j])
- {
- if(P[i].op==)
- updata(P[i].y,P[i].w);
- temp[k++]=P[i++];
- }
- else
- {
- if(P[j].op==)
- ans[P[j].id]+=P[j].w*getsum(P[j].y);
- temp[k++]=P[j++];
- }
- }
- while(i<=m)
- temp[k++]=P[i++];
- while(j<=r)
- {
- if(P[j].op==)
- ans[P[j].id]+=P[j].w*getsum(P[j].y);
- temp[k++]=P[j++];
- }
- for(i=l;i<=r;i++)
- {
- clear(P[i].y);
- P[i]=temp[i];
- }
- }
- int main()
- {
- int n,op,x1,y1,x2,y2,a;
- while(~scanf("%d%d",&n,&n))
- {
- pn=qn=maxy=;
- while(scanf("%d",&op)&&op!=)
- {
- if(op==)
- {
- scanf("%d%d%d",&x1,&y1,&a);
- x1++,y1++;
- addp(x1,y1,op,a,);
- maxy=max(maxy,y1);
- }
- else
- {
- scanf("%d%d%d%d",&x1,&y1,&x2,&y2);
- x1++,y1++,x2++,y2++;
- addp(x1-,y1-,op,,qn);
- addp(x1-,y2,op,-,qn);
- addp(x2,y1-,op,-,qn);
- addp(x2,y2,op,,qn);
- qn++;
- maxy=max(maxy,max(y1,y2));
- }
- }
- cdq(,pn-);
- for(int i=;i<qn;i++)
- {
- printf("%d\n",ans[i]);
- ans[i]=;
- }
- }
- return ;
- }
园丁都会了这个肯定会了
题目大意:有n朵花,每朵花有三个属性:花形(s)、颜色(c)、气味(m),用三个整数表示。现在要对每朵花评级,一朵花的级别是它拥有的美丽能超过的花的数量。定义一朵花A比另一朵花B要美丽,当且仅Sa>=Sb,Ca>=Cb,Ma>=Mb。显然,两朵花可能有同样的属性。需要统计出评出每个等级的花的数量。
这就是个三维偏序的问题了,没有什么更新和查询就是(s,c,m)的三维有序对求顺序对,我们同样是让sort按s由小到大,s相同按c排,c相同按m排的cmp使得第一维s有序,然后cdq分治处理第二维c,树状数组维护第三维m,最后提交,好的,wrong了。因为一个重要的点,,两朵花可能有同样的属性,如果我们不去重的话,那么相同的属性的话,因为它们的顺序不同造成它们的等级不同。所以我们把属性相同的归为一类,然后统计这一类有多少个,然后作为树状数组维护的权值维护就行,有个小细节就是当分到最小的子区间也就是只有一朵花时,它的等级应该还得提升它的权值-1,因为那些和它属性相同的合为一类了,具体的就看代码注释。
- #include<cstdio>
- #include<algorithm>
- #define lowb(x) (x&(-x))
- using namespace std;
- const int N=,K=;
- struct Flower{
- int id,s,c,m,w;
- friend bool operator == (const Flower &f1,const Flower &f2){
- return f1.s==f2.s&&f1.m==f2.m&&f1.c==f2.c;
- }
- }F[N],temp[N];
- int fn,ans[N]={},rank[N]={},summ[K]={};
- //ans[i]保存编号为i的花的等级,也就是属性小于或等于它的数目
- bool cmp(const Flower &f1,const Flower &f2){
- return f1.s==f2.s ? (f1.c==f2.c ? f1.m<f2.m : f1.c<f2.c) : f1.s<f2.s;
- }
- void updata(int m,int val)
- {
- while(m<=)
- {
- summ[m]+=val;
- m+=lowb(m);
- }
- }
- int getsum(int m)
- {
- int ans=;
- while(m)
- {
- ans+=summ[m];
- m-=lowb(m);
- }
- return ans;
- }
- void clear(int m)
- {
- while(m<=)
- {
- if(summ[m])
- summ[m]=;
- else
- break;
- m+=lowb(m);
- }
- }
- void cdq(int l,int r)
- {
- if(l==r)
- {
- ans[F[l].id]+=F[l].w-;//因为相同属性的归到一类了
- //所以还得加上F[l].w-1,也就是除它之外,
- //其他相同属性的花的数目
- return ;
- }
- int mid=(l+r)>>;
- cdq(l,mid);
- cdq(mid+,r);
- int i=l,j=mid+,k=l;
- while(i<=mid&&j<=r)
- {
- if(F[i].c<=F[j].c)
- {
- updata(F[i].m,F[i].w);
- temp[k++]=F[i++];
- }
- else
- {
- ans[F[j].id]+=getsum(F[j].m);
- temp[k++]=F[j++];
- }
- }
- while(i<=mid)
- temp[k++]=F[i++];
- while(j<=r)
- {
- ans[F[j].id]+=getsum(F[j].m);
- temp[k++]=F[j++];
- }
- for(i=l;i<=r;i++)
- {
- clear(F[i].m);
- F[i]=temp[i];
- }
- }
- int main()
- {
- int n,k;
- while(~scanf("%d%d",&n,&k))
- {
- for(int i=;i<n;i++)
- {
- F[i].w=;
- scanf("%d%d%d",&F[i].s,&F[i].c,&F[i].m);
- }
- sort(F,F+n,cmp);
- fn=;
- //去重
- for(int i=;i<n;i++)
- {
- if(i&&F[i]==F[i-])//如果和前一朵花属性相同,归到一类
- F[fn-].w++;
- else
- {
- F[fn]=F[i];
- F[fn].id=fn++;
- }
- }
- //原先n朵花去重就只有fn朵
- cdq(,fn-);
- for(int i=;i<fn;i++)
- {
- rank[ans[F[i].id]]+=F[i].w;
- ans[F[i].id]=;
- }
- for(int i=;i<n;i++)
- {
- printf("%d\n",rank[i]);
- rank[i]=;
- }
- }
- return ;
- }
公子世无双
CDQ解决一些三维偏序的问题的更多相关文章
- 并不对劲的cdq分治解三维偏序
为了反驳隔壁很对劲的太刀流,并不对劲的片手流决定与之针锋相对,先一步发表cdq分治解三维偏序. 很对劲的太刀流在这里-> 参照一.二维偏序的方法,会发现一位偏序就是直接排序,可以看成通过排序使 ...
- 【算法学习】【洛谷】cdq分治 & P3810 三维偏序
cdq是何许人也?请参看这篇:https://wenku.baidu.com/view/3b913556fd0a79563d1e7245.html. 在这篇论文中,cdq提出了对修改/询问型问题(Mo ...
- hdu5618(cdq分治求三维偏序)
题意:给n(n<=100000)个点,坐标为(xi,yi,zi)(1<=xi,yi,zi<=100000),定义一个点A比一个点B小,当且仅当xA<=xB,yA<=yB, ...
- BZOJ 3262: 陌上花开 (cdq分治,三维偏序)
#include <iostream> #include <stdio.h> #include <algorithm> using namespace std; c ...
- SPOJ:Another Longest Increasing Subsequence Problem(CDQ分治求三维偏序)
Given a sequence of N pairs of integers, find the length of the longest increasing subsequence of it ...
- BZOJ 3295:[Cqoi2011]动态逆序对(三维偏序 CDQ分治+树状数组)
http://www.lydsy.com/JudgeOnline/problem.php?id=3295 题意:简单明了. 思路:终于好像有点明白CDQ分治处理三维偏序了.把删除操作看作是插入操作,那 ...
- HDU 5517 【二维树状数组///三维偏序问题】
题目链接:[http://acm.split.hdu.edu.cn/showproblem.php?pid=5517] 题意:定义multi_set A<a , d>,B<c , d ...
- P4390 [BOI2007]Mokia 摩基亚 (CDQ解决三维偏序问题)
题目描述 摩尔瓦多的移动电话公司摩基亚(Mokia)设计出了一种新的用户定位系统.和其他的定位系统一样,它能够迅速回答任何形如"用户C的位置在哪?"的问题,精确到毫米.但其真正高科 ...
- P3157 [CQOI2011]动态逆序对 (CDQ解决三维偏序问题)
P3157 [CQOI2011]动态逆序对 题目描述 对于序列A,它的逆序对数定义为满足i<j,且Ai>Aj的数对(i,j)的个数.给1到n的一个排列,按照某种顺序依次删除m个元素,你的任 ...
随机推荐
- ccpc湘潭邀请赛 Partial Sum
选定最多m的区间,使区间和的绝对值最大.但是左右端点不能重复选取 首先涉及到区间和的问题,就应该想到用前缀和去优化 这里对前缀和排序 然后贪心的去选取最大.次大 (比赛的时候脑子堵的很,没想出来 可惜 ...
- 3.Shell的基本功能
3.Shell的基本功能Bash是Bourne-Again Shell的缩写.Bourne Shell的内部命令在Bash中同样适用.3.1 Shell语法3.1.1 Shell操作shell读取和执 ...
- Tomcat 的部署器
要使用一个Web应用程序,必须要将表示该应用程序的Context实例部署到一个Host实例中,在Tomcat中,Context实例可以用WAR文件的形式来部署,也可以将整个WEB应用程序复制到Tomc ...
- ES6入门五:箭头函数、函数与ES6新语法
箭头函数的基本用法与特点 函数与ES6新语法 一.箭头函数的基本用法与特点 声明箭头函数采用声明变量和常量的关键字:var.let.const 箭头函数的参数:没有参数用"()"空 ...
- HTML5之历史记录(实现的当页面应用路由器的底层)
history hashchange与popstate 一.history history.back():加载history列表中的前一个URL history.forward():加载history ...
- java技术面试之面试题大全
转载自:http://blog.csdn.net/lijizhi19950123/article/details/77679489 Java 面试知识点总结 本篇文章会对面试中常遇到的Java技术点进 ...
- ceres for Android 太慢的解决方法
跨平台编译了ceres,结果在android平台上运行的太慢,优化一次要0.3秒左右,时不时要一两秒.这太扯了.没辙了,在google上瞎搜索,看到 Jacobian evaluation is ve ...
- Spring Boot WebFlux整合mongoDB
引入maven文件 <dependency> <groupId>org.springframework.boot</groupId> <artifactId& ...
- Linux CentOS 7 防火墙与端口设置操作
CentOS升级到7之后用firewall代替了iptables来设置Linux端口, 下面是具体的设置方法: []:选填 <>:必填 [<zone>]:作用域(block.d ...
- Asp.Net Zero通用打印实现
Asp.Net Zero是一款非常优秀的web框架,可以用来快速构建业务系统.框架满足了业务系统所需的大部分通用功能,但是系统必须的打印报表功能一直没有实现.下面给大家介绍如何在zero中集成打印功能 ...