传送门:QAQQAQ

题意:给你一棵树,有三种操作,设所有点本来未感染

1:感染节点i,若i被二次感染,则感染i的儿子(若儿子也被感染,则感染孙子,直到到底或者感染了健康点)

2:使i子树全部健康

3:查询节点x是否被感染

思路:树上的修改查询,很容易想到树链剖分

我们先把所有点的权值设为-1,定义一个点没有感染当且仅当它的最大后缀和小于0(这主要是由操作1想到这种方法,这样对于1操作,把i节点权值+1即可)

对于2操作,在把i子树全部赋成-1时,也要消除i祖先可能有节点大于0对i子树的影响(因为全部健康后要求i子树所有点最大后缀和小于0),所以我们再次对i进行修改,使i节点权值变为$-query(Father[i])-1$,query表示i的最大后缀和,若query是-1,千万不要还是$-query-1$把i赋成了0,应该赋成-1,需要判一下,赛程上就是因为这个滑点100->30,还有若i是根节点第二个改父亲操作不要做,否则RE

代码(从树链剖分模板里HE的,码风很丑,感觉细节挺多,可能是我太弱惹QAQ):

#include<bits/stdc++.h>
using namespace std;
typedef pair<int,int> pii;
const int N=202000;
const int inf=(int)2e9; struct node{
int sum,lazy=0,lz,rz,mx;//最大后缀和
}tree[N*4];
int w[N],n,m;
int top[N],f[N],son[N],sz[N],dep[N],p[N];
int id[N],rk[N];
//rk:节点号->树剖序 id:树剖序->节点号 node operator + (node A,node B)
{
node ret;
ret.sum=A.sum+B.sum;
ret.mx=max(A.mx,A.sum+B.mx);
return ret;
} int first[N],nxt[N*2],point[N*2],e=0;
void add_edge(int x,int y)
{
point[e]=y;
nxt[e]=first[x];
first[x]=e++;
} void dfs1(int u)
{
int maxsz=0;
for(int i=first[u];i!=-1;i=nxt[i])
{
int v=point[i];
if(v==f[u]) continue;
f[v]=u; dep[v]=dep[u]+1;
dfs1(v); sz[u]+=sz[v];
if(maxsz<sz[v])
{
maxsz=sz[v];
son[u]=v;
}
}
sz[u]++;
} int cnt=0;
void dfs2(int u,int t)
{
top[u]=t;
rk[u]=++cnt;
id[cnt]=u;
if(!son[u]) return;
dfs2(son[u],t);
for(int i=first[u];i!=-1;i=nxt[i])
{
int v=point[i];
if(v==f[u]||v==son[u]) continue;
dfs2(v,v);
}
} int len(node fa)
{
return fa.rz-fa.lz+1;
} void pushup(node &fa,node ls,node rs)
{
fa.sum=ls.sum+rs.sum;
fa.lz=ls.lz; fa.rz=rs.rz;
fa.mx=max(rs.sum+ls.mx,rs.mx);
} void pushdown(node &fa,node &ls,node &rs)
{
if(!fa.lazy) return;
ls.lazy=fa.lazy; rs.lazy=fa.lazy;
ls.sum=fa.lazy*len(ls); rs.sum=fa.lazy*len(rs);
ls.mx=-1; rs.mx=-1;
fa.lazy=0;
} void build(int x,int l,int r)
{
if(l==r)
{
tree[x].sum=w[id[l]];//线段树维护的是树剖序中的第l个,而传的点是节点序,id:树剖序->节点序
tree[x].lz=l; tree[x].rz=l;
tree[x].mx=w[id[l]];
return;
}
int mid=(l+r)>>1;
build(x+x,l,mid);
build(x+x+1,mid+1,r);
pushup(tree[x],tree[x+x],tree[x+x+1]);
} void update(int x,int l,int r,int L,int R,int upd)
{
if(L<=l&&r<=R)
{
tree[x].sum=upd*len(tree[x]);
if(L!=R) tree[x].lazy=upd;
tree[x].mx=upd;
return;
}
pushdown(tree[x],tree[x+x],tree[x+x+1]);
int mid=(l+r)>>1;
if(mid>=R) update(x+x,l,mid,L,R,upd);
else if(mid<L) update(x+x+1,mid+1,r,L,R,upd);
else
{
update(x+x,l,mid,L,R,upd);
update(x+x+1,mid+1,r,L,R,upd);
}
pushup(tree[x],tree[x+x],tree[x+x+1]);
} node query(int x,int l,int r,int L,int R)
{
node ret;
ret.sum=-inf; ret.mx=-inf;
if(L<=l&&r<=R) return tree[x];
pushdown(tree[x],tree[x+x],tree[x+x+1]);
int mid=(l+r)>>1;
if(mid>=R) return query(x+x,l,mid,L,R);
else if(mid<L) return query(x+x+1,mid+1,r,L,R);
else
{
ret=query(x+x+1,mid+1,r,L,R);
ret=ret+query(x+x,l,mid,L,R);
}
return ret;
} int max_line(int x,int y)
{
node ret;
ret.sum=-inf; ret.mx=-inf;
while(top[x]!=top[y])
{
if(dep[top[x]]<dep[top[y]]) swap(x,y);
if(ret.mx==-inf) ret=query(1,1,n,rk[top[x]],rk[x]);
else ret=ret+query(1,1,n,rk[top[x]],rk[x]);
x=f[top[x]];
//不能两个都跳,可能top深度较浅的会跳过LCA
}
if(rk[x]>rk[y]) swap(x,y);
if(ret.mx==-inf) ret=query(1,1,n,rk[x],rk[y]);
else ret=ret+query(1,1,n,rk[x],rk[y]);
return ret.mx;
} void init()
{
memset(first,-1,sizeof(first));
memset(nxt,-1,sizeof(nxt));
memset(w,-1,sizeof(w));
scanf("%d%d",&n,&m);
dep[1]=1; f[1]=1;
for(int i=2;i<=n;i++)
{
scanf("%d",&p[i]);
add_edge(i,p[i]);
add_edge(p[i],i);
}
dfs1(1);
dfs2(1,1);
//for(int i=1;i<=n;i++) T[rk[i]]=w[i];//枚举节点序->树剖序
build(1,1,n);
} void solve(int opt)
{
//一个子树的dfn序是连续的
int x;
if(opt==1)
{
scanf("%d",&x);
int tmp=query(1,1,n,rk[x],rk[x]).sum+1;
update(1,1,n,rk[x],rk[x],tmp);
}
else if(opt==2)
{
scanf("%d",&x);
update(1,1,n,rk[x],rk[x]+sz[x]-1,-1);
if(x==1) return;
int tmp=min(-max_line(p[x],1)-1,-1);//!!!!!
update(1,1,n,rk[x],rk[x],tmp);
}
else
{
scanf("%d",&x);
int ans=max_line(x,1);
if(ans>=0) puts("black");
else puts("white");
}
} int main()
{
init();
for(int i=1;i<=m;i++)
{
int opt; scanf("%d",&opt);
solve(opt);
}
return 0;
}

CF1017G——The Tree的更多相关文章

  1. CF1017G The Tree 树链剖分

    CF1017G The Tree LG传送门 树链剖分好题. 乍一看还以为是道沙比题,然后发现修改操作有点不一样. 但是如果你对基本操作还不太熟练,可以看看我的树链剖分总结 有三个操作: 从一个点往下 ...

  2. CF1017G The Tree

    /* 这是什么神仙题目QAQ 首先考虑在序列上的问题 先不考虑修改成白色, 一个白点能r被染成黑色 意味着能够找到一个l使得在l-r中的操作1次数大于等于 r - l + 1 我们把初始值覆盖成-1就 ...

  3. [CF1017G]The Tree[树链剖分+线段树]

    题意 给一棵一开始 \(n\) 个点全是白色的树,以 \(1\) 为根,支持三种操作: 1.将某一个点变黑,如果已经是黑色则该操作对所有儿子生效. 2.将一棵子树改成白色. 3.询问某个点的颜色. \ ...

  4. [数据结构]——二叉树(Binary Tree)、二叉搜索树(Binary Search Tree)及其衍生算法

    二叉树(Binary Tree)是最简单的树形数据结构,然而却十分精妙.其衍生出各种算法,以致于占据了数据结构的半壁江山.STL中大名顶顶的关联容器--集合(set).映射(map)便是使用二叉树实现 ...

  5. SAP CRM 树视图(TREE VIEW)

    树视图可以用于表示数据的层次. 例如:SAP CRM中的组织结构数据可以表示为树视图. 在SAP CRM Web UI的术语当中,没有像表视图(table view)或者表单视图(form view) ...

  6. 无限分级和tree结构数据增删改【提供Demo下载】

    无限分级 很多时候我们不确定等级关系的层级,这个时候就需要用到无限分级了. 说到无限分级,又要扯到递归调用了.(据说频繁递归是很耗性能的),在此我们需要先设计好表机构,用来存储无限分级的数据.当然,以 ...

  7. 2000条你应知的WPF小姿势 基础篇<45-50 Visual Tree&Logic Tree 附带两个小工具>

    在正文开始之前需要介绍一个人:Sean Sexton. 来自明尼苏达双城的软件工程师.最为出色的是他维护了两个博客:2,000Things You Should Know About C# 和 2,0 ...

  8. Leetcode 笔记 110 - Balanced Binary Tree

    题目链接:Balanced Binary Tree | LeetCode OJ Given a binary tree, determine if it is height-balanced. For ...

  9. Leetcode 笔记 100 - Same Tree

    题目链接:Same Tree | LeetCode OJ Given two binary trees, write a function to check if they are equal or ...

随机推荐

  1. Python实现的数据结构与算法之基本搜索详解

    一.顺序搜索 顺序搜索 是最简单直观的搜索方法:从列表开头到末尾,逐个比较待搜索项与列表中的项,直到找到目标项(搜索成功)或者 超出搜索范围 (搜索失败). 根据列表中的项是否按顺序排列,可以将列表分 ...

  2. Java 异常 Failed to convert property value of type 'java.lang.String' to required type 'java.util.Date'

    查询时发送给服务器的日期的字符串格式:yyyy-MM-dd HH:mm:ss 服务器接收到日期的字符串之后,向 MySQL 数据库发起查询时,因为没有指定日期时间格式,导致字符串数据不能正确地转换为日 ...

  3. P4454 [CQOI2018]破解D-H协议

    链接 这题并不难只是需要把题读懂 - By ShadderLeave 一句话题意 给定两个数 \(p\)和\(g\),有\(t\)组询问,每组询问给出\(A\)和\(B\) 其中 A = \(g^a ...

  4. 设置 eclipse C++ 版本

    gcc 版本 4.8.5 20150623 (Red Hat 4.8.5-28) (GCC) 默认是使用 C++ 98 版本进行编译 设置 eclipse 中 C++ 的版本: Project  -& ...

  5. 你想了解的《javaScript语言精粹》(三)

    # javaScript语言精粹  # 第三章 对象 - javaScript 数据类型     1. 基础数据类型         Number String Boolean Undefined N ...

  6. 多测师讲解自动化测试 _RF自定义关键字_高级讲师肖sir

    RF自定义关键字 在rf中叫关键字 在python中就叫做函数 或实例方法 我们自己可以写自定义关键字 自己创建一个库===库里面去创建模块===模块里面创建类和实例方法==>rf导入和引用 库 ...

  7. 【编程学习笔记】如何组织构建多文件 C 语言程序!编程也有~

    优秀 Unix 程序哲学 首先,你要知道这个 C 程序是一个 Unix 命令行工具.这意味着它运行在(或者可被移植到)那些提供 Unix C 运行环境的操作系统中.当贝尔实验室发明 Unix 后,它从 ...

  8. centos8平台使用strace跟踪系统调用

    一,strace的用途 strace  是最常用的跟踪进程系统调用的工具. 说明:刘宏缔的架构森林是一个专注架构的博客,地址:https://www.cnblogs.com/architectfore ...

  9. 从Linux源码看Socket(TCP)的listen及连接队列

    从Linux源码看Socket(TCP)的listen及连接队列 前言 笔者一直觉得如果能知道从应用到框架再到操作系统的每一处代码,是一件Exciting的事情. 今天笔者就来从Linux源码的角度看 ...

  10. PyTorch常用参数初始化方法详解

    1. 均匀分布 torch.nn.init.uniform_(tensor, a=0, b=1) 从均匀分布U(a, b)中采样,初始化张量. 参数: tensor - 需要填充的张量 a - 均匀分 ...