2002: [Hnoi2010]Bounce 弹飞绵羊
时间:2019-07-20
本文章向大家介绍2002: [Hnoi2010]Bounce 弹飞绵羊,主要包括2002: [Hnoi2010]Bounce 弹飞绵羊使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。
2002: [Hnoi2010]Bounce 弹飞绵羊
时间限制: 10 Sec 内存限制: 259 MB题目描述
某天,Lostmonkey发明了一种超级弹力装置,为了在他的绵羊朋友面前显摆,他邀请小绵羊一起玩个游戏。游戏一开始,Lostmonkey在地上沿着一条直线摆上n个装置,每个装置设定初始弹力系数ki,当绵羊达到第i个装置时,它会往后弹ki步,达到第i+ki个装置,若不存在第i+ki个装置,则绵羊被弹飞。绵羊想知道当它从第i个装置起步时,被弹几次后会被弹飞。为了使得游戏更有趣,Lostmonkey可以修改某个弹力装置的弹力系数,任何时候弹力系数均为正整数。
输入
第一行包含一个整数n,表示地上有n个装置,装置的编号从0到n-1,接下来一行有n个正整数,依次为那n个装置的初始弹力系数。第三行有一个正整数m,接下来m行每行至少有两个数i、j,若i=1,你要输出从j出发被弹几次后被弹飞,若i=2则还会再输入一个正整数k,表示第j个弹力装置的系数被修改成k。对于20%的数据n,m<=10000,对于100%的数据n<=200000,m<=100000
输出
对于每个i=1的情况,你都要输出一个需要的步数,占一行。
样例输入
4
1 2 1 1
3
1 1
2 1 1
1 1
1 2 1 1
3
1 1
2 1 1
1 1
样例输出
2
3
3
算法:分块算法
思路:将n个数分成若干块,每一块是单独的一部分,互不干扰,每一块的任意节点都是直接跳到下一块的位置,这个从代码里理解,查询的时间复杂度最多也是O(sqrt(n)),这样大大减少了时间,没修改一个值的话,便更新那个值所在的那个块区域,不影响其他区域。
#include <iostream> #include <cstdio> #include <cmath> using namespace std; int a[200005]; //存取当前数组下一步走的距离 int v[200005]; //存取下一步的位置 int step[200005]; //存取当前的步数 int left1[200005]; //存取左端点的位置 int right1[200005]; //存取右端点的位置 int belong[200005]; //存取当前位置属于哪一块 int main() { int n; scanf("%d", &n); int block = sqrt(n); int num = n / block; if(n % block) { num++; } for(int i = 1; i <= num; i++) { left1[i] = (i - 1) * block + 1; right1[i] = i * block; } for(int i = 1; i <= n; i++) { scanf("%d", &a[i]); belong[i] = (i - 1) / block + 1; } //前面基本上都是模板,从这里开始变形 for(int i = n; i >= 1; i--) { v[i] = a[i] + i; //获取下一个位置 step[i] = 1; //步数初始化 if(v[i] <= n && belong[i] == belong[v[i]]) { //判断下一步的位置和当前位置是否再同一块里,如果在的话便更新状态 //注意,下面这两部分千万不能写反了,不然就会出问题,当初的我...流泪... //因为你换一下位置的话就是更新为下下步的位置了 step[i] = step[v[i]] + 1; //更新为下一步的步数 v[i] = v[v[i]]; //更新为下一步的位置 } } int m; scanf("%d", &m); while(m--) { int q, x, y; scanf("%d", &q); if(q == 1) { scanf("%d", &x); x++; //千万记住这里要自增一下,因为题目里说明了是0 ~ n-1 int ans = 0; while(x <= n) { //一块一块找过去,获取结果 ans += step[x]; x = v[x]; } printf("%d\n", ans); } else { scanf("%d %d", &x, &y); x++; //同上,自增 a[x] = y; //只需要更新改变值的那一块 for(int i = right1[belong[x]]; i >= left1[belong[x]]; i--) { v[i] = a[i] + i; step[i] = 1; if(v[i] <= n && belong[i] == belong[v[i]]) { step[i] = step[v[i]] + 1; v[i] = v[v[i]]; } } } } return 0; }
原文地址:https://www.cnblogs.com/buhuiflydepig/p/11219243.html
- J2Cache——Java两级缓存框架
- SSM三大框架整合详细总结(Spring+SpringMVC+MyBatis)
- 不到百行代码实现,类似iPhone的滑块开关
- 《深入理解Java虚拟机》(一)Java虚拟机发展史
- Servlet是如何实现MVC的?
- 基础篇章:关于 React Native 的props,state,style的讲解
- 《深入理解Java虚拟机》(三)垃圾收集器与内存分配策略
- 一个类似于进度和打卡进度的自定义view
- 《深入理解Java虚拟机》(四)虚拟机性能监控与故障处理工具
- 第七章:Shiro的Session管理——深入浅出学Shiro细粒度权限开发框架
- 《深入理解Java虚拟机》(六)堆内存使用分析,垃圾收集器 GC 日志解读
- Knockout.js是什么?
- 简单方法检测远端用户的反病毒软件
- 使用 JS 构建跨平台的原生应用(一):React Native for Android 初探
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- Redis慢查询日志
- LinkedList源码阅读笔记
- RTSP协议网络摄像头接入视频平台EasyNVR
- 在 Hiplot 中使用 Sigflow
- 0805-CDH5中的Parquet迁移至CDP中兼容性验证
- 为什么我的Redis这么“慢”?
- Nginx系列:数据压缩
- Ray,面向新兴AI应用的分布式框架
- 60.Vue export default 和 export 的使用方式
- Idea开发maven插件
- redis实战 migrate异常NOAUTH Authentication required.
- linux内存使用情况分析(free + top)
- Centos7 python3安装
- crontab JAVA_HOME not found
- Centos7 源码安装mysql5.6