Description

自从WC退役以来,大爷是越来越懒惰了。为了帮助他活动筋骨,也是受到了弹飞绵羊一题的启发,机房的小伙伴们决定齐心合力构造一个下面这样的序列。这个序列共有N项,每项都代表了一个小伙伴的力量值,如果大爷落到了第i个小伙伴的手里,那么第i个小伙伴会把大爷弹到第i+ai个小伙伴手里,其中ai就是第i个小伙伴的力量值,也就是序列的第i项。然而,因为大爷太沉了,所以有些小伙伴不能撑到锻(you)炼(xi)结束,所以我们中途会替换一些小伙伴,也就是改变序列的某些项。而且,因为大爷太沉了,所以有些小伙伴不能把大爷扔向前方,而是会把大爷往反方向扔,也就是序列中的一些项会是负的(当然,也可能是零喽)。现在机智的大爷通过在空中的观察,已经知道小伙伴们的所有活动——即初始序列、所有更改操作,他想请你算一算,如果他在某时刻落到了某个位置,那么他会在几次弹起之后落到小伙伴序列之外(毕竟摔在地上还是蛮疼的)。

Input

第一行为两个整数N和M,代表序列长度和操作次数。
第二行为N个整数,代表初始的小伙伴序列。
接下来有M行,每行代表一个操作。
如果这一行的第一个数是1,代表该操作是一个询问操作,接下来一个数X,代表询问此时大爷从X处,经过几次弹起会摔在地上。如果永远不会摔在地上,请输出-1。
如果这一行的第一个数是2,代表该操作是一个更改操作,接下来两个数X,Y,代表将序列的第X项改为Y。
N,M <= 200000  |Ai| < N

Output

对于每次询问操作,输出弹起次数或-1。

Sample Input

3 19
1 1 1
1 1
1 2
1 3
2 1 2
1 1
1 2
1 3
2 3 -1
1 1
1 2
1 3
2 2 233
1 1
1 2
1 3
2 2 -233
1 1
1 2
1 3

Sample Output

3
2
1
2
2
1
-1
-1
-1
3
1
2
3
1
2

调了两天终于A了……万分感谢czl大爷、cyc大爷和yy大爷。

RE了几发之后一直调不出来,心态爆炸开始乱搞,然后就WA了。怪我自己没有理解好写法。

因为有环,所以需要隐藏环上的一条边来使其成为森林,hl数组与hr数组即代表隐藏边的端点。

剩下的直接上代码……

 #include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int N=;
int n,m,t,u,v,a[N],hl[N],hr[N],st[N];
struct node{int fa,c[],s;bool rev;}tr[*N];
int read()
{
int x=,f=;char c=getchar();
while(c<''||c>''){if(c=='-')f=-;c=getchar();}
while(c>=''&&c<=''){x=x*+c-'';c=getchar();}
return x*f;
}
bool isroot(int k){return !k||!tr[k].fa||(tr[tr[k].fa].c[]!=k&&tr[tr[k].fa].c[]!=k);}
void up(int k){tr[k].s=tr[tr[k].c[]].s+tr[tr[k].c[]].s+;}
void down(int k)
{
int l=tr[k].c[],r=tr[k].c[];
if(tr[k].rev)
{
tr[k].rev^=;tr[l].rev^=;tr[r].rev^=;
swap(tr[k].c[],tr[k].c[]);
}
if(l)hl[l]=hl[k],hr[l]=hr[k];
if(r)hl[r]=hl[k],hr[r]=hr[k];
}
void rotate(int x)
{
int y=tr[x].fa,z=tr[y].fa,l,r;
if(tr[y].c[]==x)l=;else l=;r=l^;
if(!isroot(y)){if(tr[z].c[]==y)tr[z].c[]=x;else tr[z].c[]=x;}
tr[x].fa=z;tr[y].fa=x;tr[tr[x].c[r]].fa=y;
tr[y].c[l]=tr[x].c[r];tr[x].c[r]=y;
up(y);up(x);
}
void splay(int x)
{
int top=;st[++top]=x;
for(int i=x;!isroot(i);i=tr[i].fa)st[++top]=tr[i].fa;
for(int i=top;i;i--)down(st[i]);
while(!isroot(x))
{
int y=tr[x].fa,z=tr[y].fa;
if(!isroot(y))
{
if((tr[y].c[]==x)^(tr[z].c[]==y))rotate(x);
else rotate(y);
}
rotate(x);
}
}
void acs(int x)
{
int t=;
while(x){splay(x);tr[x].c[]=t;up(x);t=x;x=tr[x].fa;}
}
void mkroot(int x){acs(x);splay(x);tr[x].rev^=;}
int find(int x){acs(x);splay(x);while(tr[x].c[])x=tr[x].c[];return x;}
void link(int x,int y){mkroot(x);tr[x].fa=y;acs(x);splay(x);}
void cut(int x,int y)
{
mkroot(x);acs(y);splay(y);
tr[x].fa=tr[y].c[]=;up(y);
hl[x]=hr[x]=hl[y]=hr[y]=;
}
void xlink(int x,int y)
{
if(find(x)==find(y))hl[y]=x,hr[y]=y;
else link(x,y);
}
void xcut(int x,int y)
{
acs(x);splay(x);int li=hl[x],ri=hr[x];
if(hl[x]==x&&hr[x]==y)hl[x]=hl[y]=hr[x]=hr[y]=;
else
{
cut(x,y);
if(li&&ri)xlink(li,ri);//划重点!xlink而不是link。
}
}
int query(int x)
{
mkroot(n+);acs(x);splay(x);
if(hl[x]&&hr[x])return -;
return tr[x].s-;
}
int main()
{
n=read();m=read();
for(int i=;i<=n+;i++)tr[i].s=;
for(int i=;i<=n;i++)
{
a[i]=read();
if(i+a[i]<=||i+a[i]>n)link(i,n+);
else xlink(i,i+a[i]);
}
while(m--)
{
t=read();
if(t==)u=read(),printf("%d\n",query(u));
else
{
u=read();v=read();
if(u+a[u]<=||u+a[u]>n)cut(u,n+);
else xcut(u,u+a[u]);
if(u+v<=||u+v>n)link(u,n+);
else xlink(u,u+v);
a[u]=v;
}
}
return ;
}

【bzoj 4764】弹飞大爷的更多相关文章

  1. BZOJ 4764: 弹飞大爷

    4764: 弹飞大爷 Time Limit: 30 Sec  Memory Limit: 256 MBSubmit: 4  Solved: 4[Submit][Status][Discuss] Des ...

  2. BZOJ 4764: 弹飞大爷 LCT

    思路并不难,主要是细节需要注意一下. 在 lct 中,删边要写成:f[lson]=0,lson=0 (因为删 x->y 时 y 不一定是左儿子,y 只是 x 的前驱) 然后 f[lson]=ls ...

  3. 【BZOJ】4764: 弹飞大爷 LCT

    [题意]给定n个数字ai,表示大爷落到i处会被弹飞到i+ai处,弹飞到>n或<1处则落地.m次操作,修改一个ai,或询问大爷落到x处经过几次落地(或-1).n,m<=10^5,|ai ...

  4. 【LCT维护基环内向树森林】BZOJ4764 弹飞大爷

    4764: 弹飞大爷 Time Limit: 30 Sec  Memory Limit: 256 MBSubmit: 101  Solved: 52[Submit][Status][Discuss] ...

  5. 【BZOJ4764】弹飞大爷 LCT

    [BZOJ4764]弹飞大爷 Description 自从WC退役以来,大爷是越来越懒惰了.为了帮助他活动筋骨,也是受到了弹飞绵羊一题的启发,机房的小伙伴们决定齐心合力构造一个下面这样的序列.这个序列 ...

  6. BZOJ4764弹飞大爷——LCT

    题目描述 自从WC退役以来,大爷是越来越懒惰了.为了帮助他活动筋骨,也是受到了弹飞绵羊一题的启发,机房的小伙伴们 决定齐心合力构造一个下面这样的序列.这个序列共有N项,每项都代表了一个小伙伴的力量值, ...

  7. bzoj4764: 弹飞大爷

    Description 自从WC退役以来,大爷是越来越懒惰了.为了帮助他活动筋骨,也是受到了弹飞绵羊一题的启发,机房的小伙伴们 决定齐心合力构造一个下面这样的序列.这个序列共有N项,每项都代表了一个小 ...

  8. [bzoj] 2002 弹飞绵羊 || LCT

    原题 简单的LCT练习题. 我们发现对于一个位置x,他只能跳到位置x+k,也就是唯一的父亲去.加入我们将弹飞的绵羊定义为跳到了n+1,那么这就形成了一棵树.而因为要修改k,所以这颗树是动态连边的,那么 ...

  9. bzoj 2002: 弹飞绵羊 Link-Cut-Tree

    题目: Description 某天,Lostmonkey发明了一种超级弹力装置,为了在他的绵羊朋友面前显摆,他邀请小绵羊一起玩个游戏.游戏一开始,Lostmonkey在地上沿着一条直线摆上n个装置, ...

随机推荐

  1. [WC2005]双面棋盘(并查集+分治)

    题目描述 题解 唉,还是码力不行,写了一个多小时发现想错了又重构了一个多小时. 这道题意图很显然,动态维护联通块,有一个经典做法就是用LCT维护按照删除时间维护的最大生成树. 网上还有一种神奇的做法, ...

  2. 状压DP总结

    状态压缩就是将一行的状态压成一个二进制数,这个数的二进制形式反映了这一行的情况 比如0100111的意义为:这一排的第一个数没被使用,第二个被占用了,第三四个没被占用,第五六七个被占用 我们知道位运算 ...

  3. (转)java 序列化ID的作用

    序列化ID的作用: 其实,这个序列化ID起着关键的作用,它决定着是否能够成功反序列化!简单来说,java的序列化机制是通过在运行时判断类的serialVersionUID来验证版本一致性的.在进行反序 ...

  4. Echarts CPU监控 (折线仪表盘,图例混搭)

    https://blog.csdn.net/mengxiangfeiyang/article/details/44802939 CPU page   <script type="tex ...

  5. java中Comparatable接口和Comparator接口的区别

    1.不同类型的排序规则 .自然排序是什么?   自然排序是一种升序排序.对于不同的数据类型,升序规则不一样:   BigDecimal BigInteger Byte Double Float Int ...

  6. javaSE eclipse tomocat安装与配置

    ---恢复内容开始--- javaSE       下载:       第一步:百度收索jdk downlaod  下载地址:https://www.oracle.com/technetwork/ja ...

  7. 根据指定的key,将二维数组的value转换为string,适用于mysql的in查询

    function array_unique_join($arr,$param){ $utm_source_arr = array_unique(array_column($arr,$param)); ...

  8. lucene的CRUD操作Document(四)

    IndexWriter writer = new IndexWriter(Directory, IndexWriterConfig); 增加文档:writer.addDocument(); 读取文档: ...

  9. 2018acm-icpc徐州站后记

    大概是我所有比赛里面打的最自闭的比赛了,真的很不想写这篇比赛的后记. 记得我刚学习acm的时候,觉得如果每一场比赛都能拿一块铜牌就好了,只要不铁都是很开心的. 但是没想到真正拿铜会这么难受.可能是因为 ...

  10. Yearning 介绍(SQL审核平台)

    介绍 Yearning SQL 审计平台 基于Vue.js与Django的整套mysql-sql审核平台解决方案.提供基于Inception的SQL检测及执行. GitHub:https://gith ...