ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

三维偏序 模板

2019-07-10 12:53:53  阅读:193  来源: 互联网

标签:偏序 半边 nn int scanf mid 三维 模板 define


  

题目描述

有 nn 个元素,第 ii 个元素有 a_iai​、b_ibi​、c_ici​ 三个属性,设 f(i)f(i) 表示满足 a_j \leq a_iaj​≤ai​ 且 b_j \leq b_ibj​≤bi​ 且 c_j \leq c_icj​≤ci​ 的 jj 的数量。

对于 d \in [0, n)d∈[0,n),求 f(i) = df(i)=d 的数量

输入输出格式

输入格式:

 

第一行两个整数 nn、kk,分别表示元素数量和最大属性值。

之后 nn 行,每行三个整数 a_iai​、b_ibi​、c_ici​,分别表示三个属性值。

 

输出格式:

 

输出 nn 行,第 d + 1d+1 行表示 f(i) = df(i)=d 的 ii 的数量。

 

cdq分治每次计算前一半对后一半的影响。具体地, 假设三维分别是x,y,z,先按x排序。分治时每次将前半边、后半边分别按y排序。虽然现在x的顺序被打乱了,但是前半边还是都小于后半边的,所以要是只计算前半边对后半边的偏序关系,是不会受到x的影响的。维护后一半的指针i,前一半的指针j,每次将i后移一位时,若y[j]<=y[i]则不断后移j,并不断将z[j]加入树状数组。然后再查询树状数组中有多少数小于等于z[i]。 最后要清空树状数组。

 

#include<bits/stdc++.h>
using namespace std;
//input by bxd
#define rep(i,a,b) for(int i=(a);i<=(b);i++)
#define repp(i,a,b) for(int i=(a);i>=(b);--i)
#define RI(n) scanf("%d",&(n))
#define RII(n,m) scanf("%d%d",&n,&m)
#define RIII(n,m,k) scanf("%d%d%d",&n,&m,&k)
#define RS(s) scanf("%s",s);
#define ll long long
#define see(x) (cerr<<(#x)<<'='<<(x)<<endl)
#define pb push_back
#define inf 0x3f3f3f3f
#define CLR(A,v)  memset(A,v,sizeof A)
#define lson l,m,pos<<1
#define rson m+1,r,pos<<1|1
typedef pair<int,int>pii;
//////////////////////////////////
const int N=2e6+10;
int M,t[N],n,cnt,num,ans[N];
int lowbit(int i){return i&(-i);}
void add(int x,int v){for(;x<=M;x+=lowbit(x))t[x]+=v;}
int get(int x){int ans=0;for(;x;x-=lowbit(x)) ans+=t[x];return ans;}

struct node
{
    int x,y,z,w,ans;
}s[N],a[N];
bool cmp(node a,node b)
{
    return a.x==b.x?a.y==b.y?a.z<b.z:a.y<b.y:a.x<b.x;
}
bool cmpy(node a,node b)
{
    return a.y<b.y||a.y==b.y&&a.z<b.z;
}
void cbq(int l,int r)
{
    if(l==r)return ;
    int mid=(l+r)>>1;
    cbq(l,mid);cbq(mid+1,r);
    sort(a+l,a+mid+1,cmpy);sort(a+mid+1,a+r+1,cmpy);
    int j=l;
    rep(i,mid+1,r)
    {
        while(a[j].y<=a[i].y&&j<=mid)
        add(a[j].z,a[j].w),j++;
        a[i].ans+=get(a[i].z);
    }
    rep(i,l,j-1)//这里j-1 不能改成mid
    add(a[i].z,-a[i].w);
}
int main()
{
    RII(n,M);
    rep(i,1,n){RIII(s[i].x,s[i].y,s[i].z);}
    sort(s+1,s+1+n,cmp);
    num=0;
    rep(i,1,n)
    {
        num++;
        if(s[i].x!=s[i+1].x||s[i].y!=s[i+1].y||s[i].z!=s[i+1].z)
        a[++cnt]=s[i],a[cnt].w=num,num=0;
    }
    cbq(1,cnt);
    rep(i,1,cnt)
    ans[a[i].ans+a[i].w-1]+=a[i].w;
    rep(i,0,n-1)
    printf("%d\n",ans[i]);
    return 0;
}
View Code

 

标签:偏序,半边,nn,int,scanf,mid,三维,模板,define
来源: https://www.cnblogs.com/bxd123/p/11163195.html

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有