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\n",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\n",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

 

转载于:https://www.cnblogs.com/wzj-is-a-juruo/p/5306885.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值