zoukankan      html  css  js  c++  java
  • BZOJ4154: [Ipsc2015]Generating Synergy

    Description

    给定一棵以1为根的有根树,初始所有节点颜色为1,每次将距离节点a不超过l的a的子节点染成c,或询问点a的颜色
     

    Input

    第一行一个数T,表示数据组数
    接下来每组数据的第一行三个数n,c,q表示结点个数,颜色数和操作数
    接下来一行n-1个数描述2..n的父节点
    接下来q行每行三个数a,l,c
    若c为0,表示询问a的颜色
    否则将距离a不超过l的a的子节点染成c

    Output

    设当前是第i个操作,y_i为本次询问的答案(若本次操作是一个修改则y_i为0),令z_i=i*y_i,请输出z_1+z_2+...+z_q模10^9+7

    Sample Input

    1
    4 3 7
    1 2 2
    3 0 0
    2 1 3
    3 0 0
    1 0 2
    2 0 0
    4 1 1
    4 0 0

    Sample Output

    32

    HINT



    第1,3,5,7的询问的答案分别为1,3,3,1,所以答案为 1*1+2*0+3*3+4*0+5*3+6*0+7*1=32.

    数据范围:

    对于100%的数据T<=6,n,m,c<=10^5,

    1<=a<=n,0<=l<=n,0<=c<=c
     
    把一个节点映射到平面上,以DFS序位置作为第一关键字,以深度作为第二关键字。
    那么一次修改就是对一个矩形内部的点打上懒标记,KD树水水即可。
    #include<cstdio>
    #include<cctype>
    #include<queue>
    #include<cmath>
    #include<cstring>
    #include<algorithm>
    #define rep(i,s,t) for(int i=s;i<=t;i++)
    #define dwn(i,s,t) for(int i=s;i>=t;i--)
    #define ren for(int i=first[x];i;i=next[i])
    #define L T[o].lc
    #define R T[o].rc
    using namespace std;
    const int BufferSize=1<<16;
    const int inf=1e9;
    char buffer[BufferSize],*head,*tail;
    inline char Getchar() {
        if(head==tail) {
            int l=fread(buffer,1,BufferSize,stdin);
            tail=(head=buffer)+l;
        }
        return *head++;
    }
    inline int read() {
        int x=0,f=1;char c=Getchar();
        for(;!isdigit(c);c=Getchar()) if(c=='-') f=-1;
        for(;isdigit(c);c=Getchar()) x=x*10+c-'0';
        return x*f;
    }
    const int maxn=100010;
    int D,rt,pos[maxn];
    struct Node {
        int x[2],mx[2],mn[2],setv,c;
        int lc,rc,id,fa;
        bool operator < (const Node& ths) const {return x[D]<ths.x[D];}
    }P[maxn],T[maxn];
    int n,c,q,first[maxn],next[maxn],to[maxn],st[maxn],en[maxn],dep[maxn],e,ToT;
    void AddEdge(int u,int v) {
        to[++e]=v;next[e]=first[u];first[u]=e;
    }
    void dfs(int x) {
        P[x].x[0]=st[x]=++ToT;P[x].x[1]=dep[x];P[x].id=x;
        ren dep[to[i]]=dep[x]+1,dfs(to[i]);
        en[x]=ToT;
    }
    void build(int& o,int l,int r,int c) {
        o=0;
        if(l>r) return;
        int mid=l+r>>1;o=mid;D=c;
        nth_element(P+l,P+mid,P+r+1);
        T[o]=P[mid];T[o].setv=T[o].c=1;pos[P[mid].id]=mid;
        build(L,l,mid-1,c^1);build(R,mid+1,r,c^1);
        T[L].fa=T[R].fa=o;
        T[o].mn[0]=min(T[o].x[0],min(T[L].mn[0],T[R].mn[0]));
        T[o].mn[1]=min(T[o].x[1],min(T[L].mn[1],T[R].mn[1]));
        T[o].mx[0]=max(T[o].x[0],max(T[L].mx[0],T[R].mx[0]));
        T[o].mx[1]=max(T[o].x[1],max(T[L].mx[1],T[R].mx[1]));
    //    printf("%d %d %d %d %d
    ",T[o].mn[0],T[o].mx[0],T[o].mn[1],T[o].mx[1],o);
    }
    void set(int o,int val) {if(o) T[o].setv=T[o].c=val;}
    void pushdown(int o) {
        if(!T[o].setv) return;
        set(L,T[o].setv);set(R,T[o].setv);
        T[o].setv=0;
    }
    int in(int o,int x1,int x2,int y1,int y2) {return T[o].mn[0]>=x1&&T[o].mx[0]<=x2&&T[o].mn[1]>=y1&&T[o].mx[1]<=y2;}
    int out(int o,int x1,int x2,int y1,int y2) {return T[o].mn[0]>x2||T[o].mx[0]<x1||T[o].mn[1]>y2||T[o].mx[1]<y1;}
    void modify(int o,int x1,int x2,int y1,int y2,int val) {
        if(!o) return;pushdown(o);
        if(in(o,x1,x2,y1,y2)) {set(o,val);return;}
        if(out(o,x1,x2,y1,y2)) return;
        if(T[o].x[0]>=x1&&T[o].x[0]<=x2&&T[o].x[1]>=y1&&T[o].x[1]<=y2) T[o].c=val;
        if(!out(L,x1,x2,y1,y2)) modify(L,x1,x2,y1,y2,val);
        if(!out(R,x1,x2,y1,y2)) modify(R,x1,x2,y1,y2,val);
    }
    int S[maxn],top;
    void solve() {
        n=read();c=read();q=read();
        fill(first+1,first+n+1,0);e=ToT=rt=0;
        rep(i,2,n) AddEdge(read(),i);
        dfs(1);build(rt,1,n,0);
        long long res=0;
        rep(i,1,q) {
            int x=read(),l=read(),t=read(),ans=0;
            if(t) modify(rt,st[x],en[x],dep[x],dep[x]+l,t);
            else {
                int o=pos[x];
                while(o!=rt) S[++top]=T[o].fa,o=T[o].fa;
                while(top) pushdown(S[top--]);
                ans=T[pos[x]].c;
            }
            (res+=(long long)ans*i)%=1000000007;
        }
        printf("%lld
    ",res);
    }
    int main() {
        T[0].mn[0]=T[0].mn[1]=inf;
        T[0].mx[0]=T[0].mx[1]=-inf;
        dwn(T,read(),1) solve();
        return 0;
    }
    View Code
  • 相关阅读:
    Java ES api 查询例子
    leveldb学习
    Viewstamp Replication协议
    PacificA协议
    vhost架构
    数据分片方法
    常见分布式存储系统架构分析
    数据存储(B+树 vs LSM树)
    paxos算法理解
    Raft协议理解
  • 原文地址:https://www.cnblogs.com/wzj-is-a-juruo/p/5306885.html
Copyright © 2011-2022 走看看