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

  在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. 超级实用的 Java 工具类

    超级实用的 Java 工具类 在Java中,工具类定义了一组公共方法,这篇文章将介绍Java中使用最频繁及最通用的Java工具类.以下工具类.方法按使用流行度排名,参考数据来源于Github上随机选取 ...

  2. tensorflow零起点快速入门(5) --强化学习摘录截图

    tf.random_normal_initializer tf的GraphKeys用法 tf.reduce_mean tf.squared_difference 非tf中的zip,python的zip ...

  3. Java lesson19homework

    package com.xt.lesson19; /** * 已知如下: 下表为某班级四次考试成绩单, 1. 要求使用HashMap<String, Integer>存储每次考试的成绩(k ...

  4. yolov3应该什么时候停止训练?

    按照训练期间的参数提示: Region Avg IOU:0.798363,Class:0.893232,Obj:0.700808,No Obj:0.004567,Avg Recall:1.000000 ...

  5. centos 配置rsync+inotify数据实时同步

    何为rsync? 定义: rsync是一个开源的快速备份工具,可以在不同主机之间镜像同步整个目录树,支持增量备份,保持链接和权限,非常适用于异地备份 何为源端和发起端? 在远程同步过程中,负责发起rs ...

  6. git 的用法和命令

    学无止境,精益求精! 十年河东,十年河西,莫欺少年穷! 学历代表你的过去,能力代表你的现在,学习代表你的将来! 很久没写博客了,都是工作太忙闹的,索性今儿转发一篇!省的博客园太冷清了... Git图形 ...

  7. 原生JS-实现轮播图

    用原生JS实现一个轮播图(效果) HTML <div id="outer"> <ul id="imgList"> <!-- 图片列 ...

  8. 如何使用koa搭建一个简单服务

    1.首先检测是否已经有node环境?   把Windows的黑窗体的命令行工具调用出来   敲击命令行node -v , 然后,就可以看到这个打印出了一个版本号,这就证明我们的node.js已经是安装 ...

  9. 编辑docker容器中的文件

    一般docker中没有VI或者其它相应的文本编辑器,为了写个东西安装个vi就可以解决问题,除此之外还有别的办法 登陆docker中找到需要编辑的文件的位置 sudo docker ps -a sudo ...

  10. [LeetCode]1089. Duplicate Zeros

    Given a fixed length array arr of integers, duplicate each occurrence of zero, shifting the remainin ...