本来几天前就该记录的东西,硬生生被我拖到了现在,太懒了。。。

  在cdq学习时,二维偏序已经解决了,无非就是先sort使第一维有序,然后再用cdq或者数据结构处理第二维。而三维偏序的时候呢,大佬的做法好像有树套树之类的,但是我不会,所以我选择cdq。

  大体的思路很简单,首先也是先使第一维有序,然后cdq把第二维由小到大合并,这时再套个线段树或者树状数组处理第三维。来几题做一下

BZOJ1935园丁的烦恼

  题目大意:一个花园里有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 ;
}

园丁不烦恼了

BZOJ1176Mokia

  题目大意:维护一个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 ;
}

园丁都会了这个肯定会了

BZOJ3262陌上花开

  题目大意:有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解决一些三维偏序的问题的更多相关文章

  1. 并不对劲的cdq分治解三维偏序

    为了反驳隔壁很对劲的太刀流,并不对劲的片手流决定与之针锋相对,先一步发表cdq分治解三维偏序. 很对劲的太刀流在这里->  参照一.二维偏序的方法,会发现一位偏序就是直接排序,可以看成通过排序使 ...

  2. 【算法学习】【洛谷】cdq分治 & P3810 三维偏序

    cdq是何许人也?请参看这篇:https://wenku.baidu.com/view/3b913556fd0a79563d1e7245.html. 在这篇论文中,cdq提出了对修改/询问型问题(Mo ...

  3. hdu5618(cdq分治求三维偏序)

    题意:给n(n<=100000)个点,坐标为(xi,yi,zi)(1<=xi,yi,zi<=100000),定义一个点A比一个点B小,当且仅当xA<=xB,yA<=yB, ...

  4. BZOJ 3262: 陌上花开 (cdq分治,三维偏序)

    #include <iostream> #include <stdio.h> #include <algorithm> using namespace std; c ...

  5. 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 ...

  6. BZOJ 3295:[Cqoi2011]动态逆序对(三维偏序 CDQ分治+树状数组)

    http://www.lydsy.com/JudgeOnline/problem.php?id=3295 题意:简单明了. 思路:终于好像有点明白CDQ分治处理三维偏序了.把删除操作看作是插入操作,那 ...

  7. HDU 5517 【二维树状数组///三维偏序问题】

    题目链接:[http://acm.split.hdu.edu.cn/showproblem.php?pid=5517] 题意:定义multi_set A<a , d>,B<c , d ...

  8. P4390 [BOI2007]Mokia 摩基亚 (CDQ解决三维偏序问题)

    题目描述 摩尔瓦多的移动电话公司摩基亚(Mokia)设计出了一种新的用户定位系统.和其他的定位系统一样,它能够迅速回答任何形如"用户C的位置在哪?"的问题,精确到毫米.但其真正高科 ...

  9. P3157 [CQOI2011]动态逆序对 (CDQ解决三维偏序问题)

    P3157 [CQOI2011]动态逆序对 题目描述 对于序列A,它的逆序对数定义为满足i<j,且Ai>Aj的数对(i,j)的个数.给1到n的一个排列,按照某种顺序依次删除m个元素,你的任 ...

随机推荐

  1. docker入门到放弃

    1.容器简介 Docker是一个开源的应用容器引擎,使用Go语言开发,基于Linux内核的cgroup,namespace,Union FS等技术,对应用进程进行封装隔离,并且独立于宿主机与其他进程, ...

  2. URI解析

    这里主要参考 RFC3986 文档. URI可以分为URL,URN或同时具备locators 和names特性的一个东西.URN作用就好像一个人的名字,URL就像一个人的地址.换句话说:URN确定了东 ...

  3. nginx-host

    下载nginx镜像 docker pull docker.io/nginx:latest 由于calico网络不支持http协议,所以即使你在iptables中配置了nat路由,将访问宿主机80端口的 ...

  4. Java 集合和泛型

    一.集合(Collections) Java使用集合来组织和管理对象. 1.Java的集合类 集合类主要负责保存.盛装和管理对象,因此集合类也被称为容器类. 集合类分为Set.List.Map和Que ...

  5. Spring 的 Bean 管理(注解方式)

    Spring 的 Bean 管理(注解方式) 1. 导入必要的 jar 包和 xml 文件 使用注解需要导入 spring-aop 的 jar 包. applicationContext.xml 文件 ...

  6. TreeSet——实现Comparable接口并重写CompareTo()方法

    TreeSet是以自然顺序存的数据,例如 Set<Student> students=new TreeSet(); students.add(new Student("111&q ...

  7. 不同格式的YUV 和 RGB互转

    YUV色彩空间:        Y是亮度值,也就是说8位的灰度值即可组成一幅黑白图像,黑白电视机就是这样的.        UV是色彩值,是给Y上色用的.U是Cb也就是RGB中的蓝色分量,V是Cr也就 ...

  8. mysql的锁机制,以及乐观锁,悲观锁,以及热点账户余额问题

    mysql的简单锁机制. myisam 1.只支持表级锁,所以经常更新的表结构不适宜用. 2.select也会产生锁表 innodb 1.支持事务,行级锁,表级锁,执行行级锁的前提是sql语句的索引有 ...

  9. 制作linux云主机镜像

    目录 制作linux云主机镜像 1.物理机环境准备 2.安装kvm虚拟机 3.操作虚拟机 4.在物理机上处理镜像 5.拷贝制作好的raw格式的镜像 6.发布镜像到云平台 制作linux云主机镜像 1. ...

  10. STM32WB SRAM2

    SRAM2存储: 1.挂接总线及地址大小 2.地址镜像 3.RDP(read protection)等级 4.不同等级下的访问状态 5.声明位于SRAM2区中的数据 1)在icf文件中定义region ...