Description

给 \(n\) 个点以及它们的弹力系数 \(k_i\) ,含义为 可以弹到 \(i + k_i\) 的位置。

支持两个东西,修改一个点的弹力系数;求一个点要弹多少次弹出 \(n\)

Solution

用 LCT 做。弹力系数是 \(k_i\) 可以看作是 \(i\) 和 \(i+k_i\) 连了一条边。如果弹出去了就不妨设和 \(0\) 连一条边。

对于修改操作,先把原来的边删除,修改 k 数组,再连上新边

对于查询操作,维护一个子树大小 siz (这里是 splay 上的 siz,不是原树上的),然后询问就相当于当前这个点 \(u\) 到 \(0\) 这条链上有几个点。所以就 split 出来这条链然后输出 siz - 1 就行了(注意要减 \(1\) 因为问的是弹多少次)

然后就做完了(注意输入的时候要加 1)

Code

#include <bits/stdc++.h>
using namespace std;
const int N = 200100;
int n, m, K[N];
struct node {
int siz, rev;
node *ch[2], *prt;
int dir() { return prt->ch[1] == this; }
int isr() { return !prt || (prt->ch[0] != this && prt->ch[1] != this); }
void setc(node *p, int k) { this->ch[k] = p; if(p) p->prt = this; }
void upd() { int s = 1; if(ch[0]) s += ch[0]->siz; if(ch[1]) s += ch[1]->siz; siz = s; }
void push() { if(!rev) return ; swap(ch[0], ch[1]);
if(ch[0]) ch[0]->rev ^= 1; if(ch[1]) ch[1]->rev ^= 1; rev = 0; }
}*P[N], pool[N], *cur = pool; node *sta[N]; int top;
node *New() { node *p = cur++; p->siz = 1; return p; }
void rotate(node *p) {
node *prt = p->prt; int k = p->dir();
if(!prt->isr()) prt->prt->setc(p, prt->dir());
else p->prt = prt->prt; prt->setc(p->ch[!k], k);
p->setc(prt, !k); prt->upd(); p->upd();
}
void splay(node *p) {
node *q = p;
while(1) { sta[++top] = q; if(q->isr()) break; q = q->prt; }
while(top) sta[top--]->push();
while(!p->isr()) {
if(p->prt->isr()) rotate(p);
else if(p->dir() == p->prt->dir()) rotate(p->prt), rotate(p);
else rotate(p), rotate(p);
} p->upd();
}
node *access(node *p) { node *q = 0; for(; p; p = p->prt) splay(p), p->ch[1] = q, (q = p)->upd(); return q; }
void mkroot(node *p) { access(p); splay(p); p->rev ^= 1; p->push(); }
void split (node *p, node *q) { mkroot(p); access(q); splay(p); }
void link (node *p, node *q) { mkroot(p); mkroot(q); p->setc(q, 1); }
void cut (node *p, node *q) { split(p, q); p->ch[1] = q->prt = 0; }
int main() {
scanf("%d", &n); P[0] = New();
for(int i = 1; i <= n; i++) scanf("%d", &K[i]), P[i] = New();
for(int i = 1; i <= n; i++) {
if(i + K[i] <= n) link(P[i], P[i + K[i]]);
else link(P[i], P[0]);
} int m; scanf("%d", &m);
for(int i = 1; i <= m; i++) {
int op, u; scanf("%d %d", &op, &u); u++;
if(op == 1) {
split(P[0], P[u]); printf("%d\n", P[0]->siz - 1);
}
if(op == 2) { int k; scanf("%d", &k);
if(u + K[u] <= n) cut(P[u], P[u + K[u]]);
else cut(P[u], P[0]); K[u] = k;
if(u + K[u] <= n) link(P[u], P[u + K[u]]);
else link(P[u], P[0]);
}
}
return 0;
}

最新文章

  1. Android快乐贪吃蛇游戏实战项目开发教程-05虚拟方向键(四)四个三角形按钮
  2. 从外部浏览开启app
  3. java 23 - 3 单例模式实现Runtime类
  4. 三层架构实例 VB.NET版
  5. android 开发解密时出现pad block corrupted 错误
  6. 转载 https协议和http协议的区别
  7. 变身windows达人,用运行命令直接启动所有应用程序
  8. AppDelegate关于应用程序挂起、复原与终止的代理方法
  9. 警告: [SetPropertiesRule]{Server/Service/Engine/Host/Context}
  10. [ios2] 利用钥匙串,在应用里保存用户密码的方法 【转】
  11. java实现邮箱找密码
  12. 基于docker 部署 canvas-lms
  13. 【NOI2019模拟】搬砖
  14. [MicroPython]TPYBoard v102炫彩跑马灯WS2812B
  15. react-踩坑记录——页面底部多出一倍高度的空白
  16. GitHub Desktop的简单使用
  17. 1.3 Java中的标识符和关键字
  18. 什么是新生代 GC 和老年代 GC
  19. 20172329 2018-2019《Java程序设计与数据结构》课程总结
  20. SQL: 查找空值

热门文章

  1. springboot通过http访问——修改访问的端口号
  2. 解决maven update project 后项目jdk变成1.5
  3. ubuntu16.04安装cuda8.0试错锦集
  4. 【DL.AI】《Structuring Machine Learning Projects》笔记
  5. PAT 1147 Heaps
  6. Asp.net MVC area
  7. CSS 绝对定位与弹性布局合作居中
  8. ACM数论之旅14---抽屉原理,鸽巢原理,球盒原理(叫法不一又有什么关系呢╮(╯▽╰)╭)
  9. Python中pip install MySQL-python报错解决方法
  10. 计算机网络【10】—— Cookie与Session