• 三维偏序 模板


      

    题目描述

    有 nn 个元素,第 ii 个元素有 a_iaib_ibic_ici 三个属性,设 f(i)f(i) 表示满足 a_j leq a_iajai 且 b_j leq b_ibjbi 且 c_j leq c_icjci 的 jj 的数量。

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

    输入输出格式

    输入格式:

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

    之后 nn 行,每行三个整数 a_iaib_ibic_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
    ",ans[i]);
        return 0;
    }
    View Code
  • 相关阅读:
    mysql高级排序&高级匹配查询示例
    为什么我说我会这会那,但真正到回答实际问题的时候却傻了眼了?
    微软职位内部推荐-SW Engineer II for WinCE
    微软职位内部推荐-Senior Program Manager
    微软职位内部推荐-Software Engineer II-Web app
    微软职位内部推荐-Senior Software Engineer -Web
    微软职位内部推荐-Senior Software Engineer-News
    微软职位内部推荐-Software Engineer II-News
    微软职位内部推荐-SDE II
    微软职位内部推荐-Software Engineer II-Data Mini
  • 原文地址:https://www.cnblogs.com/bxd123/p/11163195.html
Copyright © 2020-2023  润新知