【题意概述】

  给一棵以1为根的树,树上的每个节点有一个ai值,代表它可以传送到自己的ai倍祖先,如果不存在则传送出这棵树。现在询问某个节点传送出这棵树需要多少步。

【题解】

  其实是把“弹飞绵羊”那道题从序列上搬到了树上,解法其实类似。

  我们可以用LCT维护传送的关系,若点i存在ai倍祖先,那么就把他们link起来,否则就把i与特殊节点n+1给link起来。

  询问某个点要传送多少次时,就是询问这个点到n+1有多远,我们在LCT上取出这一段,查询size即可。

 #include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#define N (100010)
#define ls (c[u][0])
#define rs (c[u][1])
#define LL long long
#define rg register
using namespace std;
int T,n,m,opt,cnt,tot,last[N],k[N],dfn[N],dep[N],p[N][];
struct edge{
int to,pre;
}e[N];
char buf[],*ptr=buf-;
template<typename T>
inline void read(T &k)
{
int f=; k=; char c=*++ptr;
while(c<'' || c>'') c=='-'&&(f=-), c=*++ptr;
while(c<='' && c>='') k=k*+c-'', c=*++ptr;
k*=f;
} struct Link_Cut_Tree{
int top,c[N][],fa[N],rev[N],size[N],q[N];
inline void clear(){
for(rg int i=;i<=n+;i++) c[i][]=c[i][]=size[i]=rev[i]=fa[i]=;
}
inline void pushdown(int u){
if(rev[u]) rev[ls]^=,rev[rs]^=,rev[u]^=,swap(ls,rs);
}
inline void pushup(int u){
size[u]=;
if(ls) size[u]+=size[ls];
if(rs) size[u]+=size[rs];
}
inline bool isroot(int u){
return c[fa[u]][]!=u&&c[fa[u]][]!=u;
}
inline bool which(int u){
return c[fa[u]][]==u;
}
void rotate(int u){
int f=fa[u],gf=fa[f],wh=which(u);
if(!isroot(f)) c[gf][which(f)]=u;
fa[u]=gf; fa[f]=u; fa[c[u][wh^]]=f;
c[f][wh]=c[u][wh^]; c[u][wh^]=f;
pushup(f); pushup(u);
}
void splay(int u){
q[top=]=u;
for(int i=u;!isroot(i);i=fa[i]) q[++top]=fa[i];
for(int i=top;i;i--) pushdown(q[i]);
while(!isroot(u)){
if(!isroot(fa[u])) rotate(which(u)==which(fa[u])?fa[u]:u);
rotate(u);
}
pushup(u);
}
void access(int u){
for(int son=;u;son=u,u=fa[u])
splay(u),c[u][]=son,pushup(u);
}
void makeroot(int u){
access(u); splay(u); rev[u]^=;
}
int find(int u){
access(u); splay(u);
while(ls) u=ls; splay(u);
return u;
}
void split(int x,int y){
makeroot(x); access(y); splay(y);
}
void cut(int x,int y){
int xrt=find(x),yrt=find(y);
if(xrt!=yrt) return;
split(x,y);
if(c[y][]==x) c[y][]=,fa[x]=;
pushup(y);
}
void link(int x,int y){
int xrt=find(x),yrt=find(y);
if(xrt==yrt) return;
makeroot(x); fa[x]=y;
}
}t;
void dfs(int x,int fa){
dfn[x]=++cnt; dep[x]=dep[fa]+; p[x][]=fa;
int tmp=log2(dep[x]);
for (int i=;i<=tmp;i++) p[x][i]=p[p[x][i-]][i-];
for(rg int i=last[x];i;i=e[i].pre) dfs(e[i].to,x);
}
inline int anc(int x,int y){
for(rg int i=;i<=;i++) if(y&(<<i)) x=p[x][i];
return x;
}
inline void Pre(){
for(rg int i=;i<=n;i++) last[i]=;
cnt=tot=;
t.clear();
}
int main(){
fread(buf, , sizeof(buf), stdin);
read(T);
while(T--){
read(n);
Pre();
for(rg int i=;i<=n;i++){
int fa; read(fa);
e[++tot]=(edge){i,last[fa]}; last[fa]=tot;
}
dfs(,);
// for(rg int i=1;i<=n;i++) printf("%d ",dep[i]); puts("dep");
for(rg int i=;i<=n;i++){
read(k[i]);
if(k[i]>=dep[i]) t.link(dfn[i],n+);
else t.link(dfn[i],dfn[anc(i,k[i])]);
}
read(m);
while(m--){
int opt; read(opt);
if(opt==){
int x; read(x);
x=dfn[x];
t.makeroot(x); t.access(n+); t.splay(n+);
printf("%d\n",t.size[n+]-);
}
else{
int x,y; read(x); read(y);
if(k[x]>=dep[x]&&y>=dep[x]) continue;
if(k[x]>=dep[x]) t.cut(dfn[x],n+);
else t.cut(dfn[x],dfn[anc(x,k[x])]);
k[x]=y;
if(k[x]>=dep[x]) t.link(dfn[x],n+);
else t.link(dfn[x],dfn[anc(x,k[x])]);
}
}
}
return ;
}

ACM多校联赛7 2018 Multi-University Training Contest 7 1009 Tree的更多相关文章

  1. 2018牛客网暑假ACM多校训练赛(第三场)G Coloring Tree 计数,bfs

    原文链接https://www.cnblogs.com/zhouzhendong/p/NowCoder-2018-Summer-Round3-G.html 题目传送门 - 2018牛客多校赛第三场 G ...

  2. 牛客网暑期ACM多校训练营(第三场)G:Coloring Tree(函数的思想)

    之前两次遇到过函数的思想的题,所以这次很敏感就看出来了.可以参考之前的题: https://www.cnblogs.com/hua-dong/p/9291507.html Christmas is c ...

  3. hdu 6394 Tree (2018 Multi-University Training Contest 7 1009) (树分块+倍增)

    链接: http://acm.hdu.edu.cn/showproblem.php?pid=6394 思路:用dfs序处理下树,在用分块,我们只需要维护当前这个点要跳出这个块需要的步数和他跳出这个块去 ...

  4. 2018 Nowcoder Multi-University Training Contest 2

    目录 Contest Info Solutions A. run D. monrey G. transform H. travel I. car J. farm Contest Info Practi ...

  5. 2018 Nowcoder Multi-University Training Contest 1

    Practice Link J. Different Integers 题意: 给出\(n\)个数,每次询问\((l_i, r_i)\),表示\(a_1, \cdots, a_i, a_j, \cdo ...

  6. 2018 Nowcoder Multi-University Training Contest 5

    Practice Link A. gpa 题意: 有\(n\)门课程,每门课程的学分为\(s_i\),绩点为\(c_i\),要求最多删除\(k\)门课程,使得gpa最高. gpa计算方式如下: \[ ...

  7. 2018 Nowcoder Multi-University Training Contest 10

    Practice Link J. Rikka with Nickname 题意: 给出\(n\)个字符串,要求依次合并两个串\(s, t\),满足将\(t\)合并到\(s\)中变成\(r\),使得\( ...

  8. 2018牛客网暑期ACM多校训练营(第二场)I- car ( 思维)

    2018牛客网暑期ACM多校训练营(第二场)I- car 链接:https://ac.nowcoder.com/acm/contest/140/I来源:牛客网 时间限制:C/C++ 1秒,其他语言2秒 ...

  9. 2015 HDU 多校联赛 5363 Key Set

    2015 HDU 多校联赛 5363 Key Set 题目: http://acm.hdu.edu.cn/showproblem.php? pid=5363 依据前面给出的样例,得出求解公式 fn = ...

随机推荐

  1. oracle 统计/分析函数

    Oracle从8.1.6开始提供分析函数,分析函数用于计算基于组的某种聚合值,它和聚合函数的不同之处是对于每个组返回多行,而聚合函数对于每个组只返回一行. 语法: Sql代码 <analytic ...

  2. 使用vue-cli启动项目出错

    Vue.js(读音 /vjuː/, 类似于 view) 是一套构建用户界面的渐进式框架. Vue 只关注视图层, 采用自底向上增量开发的设计. Vue 的目标是通过尽可能简单的 API 实现响应的数据 ...

  3. vue watcher

    观察 Watchers 虽然计算属性在大多数情况下更合适,但有时也需要一个自定义的 watcher .这是为什么 Vue 提供一个更通用的方法通过watch 选项,来响应数据的变化.当你想要在数据变化 ...

  4. uva10655

    Given the value of a+b and ab you will have to find the value of a n + b n Input The input file cont ...

  5. ubuntu 14.04中: 像ubuntu16.04 一样可以在文件夹内打开此路径下的shell

    sudo apt-get install nautilus-open-terminal 然后重启 ok!

  6. bzoj 3156: 防御准备【斜率优化dp】

    就是套路咯,设s[i]为1+2+...i 首先列出dp方程\( f[i]=min(f[j]+a[i]+(i-j)*i-(s[i]-s[j])) \) 然后推一推 \[ f[i]=f[j]+a[i]+( ...

  7. less新手入门(五)—— CssGuards、循环、合并

    九. CssGuards 警卫也可以应用于css选择器,这是一种语法糖,用于声明mixin,然后立即调用它. 例如,在1.5.0之前,您必须这样做 .my-optional-style() when ...

  8. Spring Boot (28) actuator与spring-boot-admin

    在上一篇中,通过restful api的方式查看信息过于繁琐,也不直观,效率低下.当服务过多的时候看起来就过于麻烦,每个服务都需要调用不同的接口来查看监控信息. SBA SBA全称spring boo ...

  9. 重新学习Java——对象和类(二)

    上一节回归了如何以面向对象的思想去使用一些Java中的公共类,也设计了一些自己的类并介绍了设计类的基本方法和技巧,这一节我们将继续回顾这些内容,并争取从中获得新的体验和感受. 1. 静态域与静态方法 ...

  10. CF842C Ilya And The Tree

    思路: 1. 如果根节点是0,那么可以通过一次dfs计算出所有节点的最大值. 2. 如果根节点不是0,那么其余各点的最大值一定是根节点的一个因子.首先计算出根节点的所有因子.在dfs到一个深度为d的节 ...