Brief description

某天,Lostmonkey发明了一种超级弹力装置,为了在他的绵羊朋友面前显摆,他邀请小绵羊一起玩个游戏。游戏一开始,Lostmonkey在地上沿着一条直线摆上n个装置,每个装置设定初始弹力系数ki,当绵羊达到第i个装置时,它会往后弹ki步,达到第i+ki个装置,若不存在第i+ki个装置,则绵羊被弹飞。绵羊想知道当它从第i个装置起步时,被弹几次后会被弹飞。为了使得游戏更有趣,Lostmonkey可以修改某个弹力装置的弹力系数,任何时候弹力系数均为正整数。

Algorithm design

因为太弱不会用lct,所以我们使用分块做Orz

考察暴力解法:

第一种暴力做法就是修改的时候直接\(\Theta(1)\)修改,查询的时候要\(\Theta(n)\)查询。

第二种暴力做法就是修改的时候把所有的答案重新递推出来,复杂度\(\Theta(n)\),查询的时候就可以\(\Theta(1)\)查询了。

我们考虑分块把两种方法结合起来。设分块的大小为\(h(n)\)。

首先我们预处理出来每个点跳出当前块需要多少步,并且跳出之后会落在哪里。

考虑修改。对于每个点,如果我们修改,会导致所有可以到达它的点的答案变化。我们只修改块内的答案,不难证明这样做是可行的。复杂度\(\Theta(h(n))\)

考虑查询。根据预处理信息,我们可以方便的查询。

Code

#include <cmath>
#include <cstdio>
#include <cstring>
const int maxn = 200100;
const int inf = 0x3f3f3f;
int n, m, f[maxn], g[maxn], h[maxn], k[maxn], block;
int id(int x) {
if (block != 0)
return (x - 1) / block + 1;
return 1;
}
int main(int argc, char const *argv[]) {
#ifndef ONLINE_JUDGE
freopen("input", "r", stdin);
#endif
scanf("%d", &n);
block = (int)sqrt(n);
memset(f, -1, sizeof(f));
for (int i = 1; i <= n; i++) {
int x;
scanf("%d", &x);
f[i] = i + x;
if (f[i] > n)
f[i] = -1;
k[i] = -1;
h[i] = -1;
g[i] = inf;
}
k[n + 1] = 0;
for (int i = n; i >= 1; i--) {
if (f[i] != -1) {
k[i] = k[f[i]] + 1;
if (id(i) == id(f[i])) {
g[i] = g[f[i]] + 1;
h[i] = h[f[i]];
} else {
g[i] = 1;
h[i] = f[i];
}
} else {
k[i] = 1;
g[i] = 1;
h[i] = -1;
}
}
scanf("%d", &m);
while (m--) {
int opt;
scanf("%d", &opt);
if (opt == 1) {
if (block == 0) {
printf("0\n");
continue;
}
int x;
scanf("%d", &x);
x++;
int ans = 1;
while (f[x] != -1) {
if (h[x] != -1) {
ans += g[x];
x = h[x];
} else {
x = f[x];
ans++;
}
}
printf("%d\n", ans);
}
if (opt == 2) {
if (block == 0)
continue;
int x, y;
scanf("%d %d", &x, &y);
x++;
f[x] = x + y;
k[x] = k[y] + 1;
if (f[x] > n)
f[x] = -1;
if (id(y) == id(x))
g[x] = g[y] + 1, h[x] = h[y];
else
g[x] = y, h[x] = 1;
for (int i = x; id(i) == id(x) && i; i--) {
if (f[i] != -1) {
k[i] = k[f[i]] + 1;
if (id(i) == id(f[i])) {
g[i] = g[f[i]] + 1;
h[i] = h[f[i]];
} else {
g[i] = 1;
h[i] = f[i];
}
} else {
k[i] = 1;
g[i] = 1;
h[i] = -1;
}
}
}
}
return 0;
}

[bzoj2002][Hnoi2010]Bounce弹飞绵羊——分块的更多相关文章

  1. bzoj2002: [Hnoi2010]Bounce 弹飞绵羊 [分块][LCT]

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

  2. bzoj2002 [Hnoi2010]Bounce 弹飞绵羊——分块

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2002 第一次用分块,感觉超方便啊: 如果记录每个点的弹力系数,那么是O(1)修改O(n)查询 ...

  3. bzoj2002: [Hnoi2010]Bounce 弹飞绵羊 分块

    这个题体现了分块不只是最大值最小值众数次数,而是一种清真的思想. 我们把整个序列分块,在每个块里处理每个位置跳出这个块的次数和跳出的位置,那么每次修改n0.5,每次查询也是,那么O(m* n0.5)的 ...

  4. BZOJ2002 Hnoi2010 Bounce 弹飞绵羊 【LCT】【分块】

    BZOJ2002 Hnoi2010 Bounce 弹飞绵羊 Description 某天,Lostmonkey发明了一种超级弹力装置,为了在他的绵羊朋友面前显摆,他邀请小绵羊一起玩个游戏.游戏一开始, ...

  5. 【bzoj2002】[Hnoi2010]Bounce 弹飞绵羊 分块

    [bzoj2002][Hnoi2010]Bounce 弹飞绵羊 2014年7月30日8101 Description 某天,Lostmonkey发明了一种超级弹力装置,为了在他的绵羊朋友面前显摆,他邀 ...

  6. BZOJ 2002: [Hnoi2010]Bounce 弹飞绵羊 分块

    2002: [Hnoi2010]Bounce 弹飞绵羊 Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOn ...

  7. [bzoj2002][Hnoi2010]Bounce弹飞绵羊_LCT

    Bounce弹飞绵羊 bzoj-2002 Hnoi-2010 题目大意:n个格子,每一个格子有一个弹簧,第i个格子会将经过的绵羊往后弹k[i]个,达到i+k[i].如果i+k[i]不存在,就表示这只绵 ...

  8. 【BZOJ2002】 [Hnoi2010]Bounce 弹飞绵羊 分块/LCT

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

  9. bzoj2002 [Hnoi2010]Bounce 弹飞绵羊【分块】

    传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=2002 这一题除了LCT解法,还有一种更巧妙,代码量更少的解法,就是分块.先想,如果仅仅记录每 ...

随机推荐

  1. LeetCode高频题目(100)汇总-Java实现

    LeetCode高频题目(100)汇总-Java实现       LeetCode高频题目(100)汇总-Java实现 目录 第01-50题 [Leetcode-easy-1] Two Sum [Le ...

  2. Python初步

    准备在工作之余看看Python的东西 收录一些资料 Python初学者(零基础学习Python.Python入门)常见问题:书籍推荐.资料.社区 http://blog.csdn.net/xiaowa ...

  3. ant-design 实现一个登陆窗口

    前提:已经完成项目实战(https://ant.design/docs/react/practical-projects-cn#定义-Model) 如果要想实现一个登陆窗口,首先得有一个ui,想到的是 ...

  4. 「个人训练」Copying Books(UVa714)

    好久不更新主要是怠惰了....还要加强训练. 题意分析与思路 注意到这样一句话: our goal is to minimize the maximum number of pages assigne ...

  5. Python 3基础教程28-内置函数

    本文介绍Python中的内置函数,Python中有很多内置的,功能强大的函数,可以帮我们解决很多问题,有些方法,根本不需要你去再次编写实现函数,你直接调用就可以.在这之前,需要介绍下,如何在windo ...

  6. Cassandra - Insert after Delete fails silently

    在delete一条数据后,再insert 相同内容的数据,结果看起来是成功的,但是当你去查找这个数据,却没有任何内容,整个过程并且没有任何异常提示. 这往往发生在单元测试的时候,我们反复清理和写入数据 ...

  7. java设计模式之装饰器模式以及在java中作用

    在JAVA I/O类库里有很多不同的功能组合情况,这些不同的功能组合都是使用装饰器模式实现的,下面以FilterInputStream为例介绍装饰器模式的使用  FilterInputStream和F ...

  8. 配置Android应用开发环境

    详解JRE.JDK.SDK.ADT请点击辨析ADK&JVM&JRE&JDK&ADT 一.安装JDK 开发 Android应用程序的时候,仅有Java运行环境(Java ...

  9. Log4Net讲解

    声明:本文内容主要译自Nauman Leghari的Using log4net,亦加入了个人的一点心得(节3.1.4). 1           简介 1.1          Log4net的优点: ...

  10. ActiveMQ使用详解---相关概念

    一.前言 公司之前使用activeMQ做过一款用于系统之间传递信息的工具,最近才正式投入生产使用,这期间出现了一些比较奇怪的问题,最终发现是没有清晰的了解activeMQ的相关配置以及一些相关概念,借 ...