题目链接

洛谷P4739

题目翻译:

你正在模拟无人机探索一个不稳定的环状行星的过程。技术上说,无人机正在穿过一个环形网格———一个在两维上都首尾环绕在一起的矩形网格。格子的行号从上到下依次编号为\(1\)到\(r\),列号

从上到下依次编号为\(1\)到\(c\)。每个格子还有一个海拔——这是个正数。

无人机一开始位于第一行第一列的格子。每一步,无人机会考虑这样三个格子:右边、右上方、右下方(注意这个网格首尾相接)。无人机会飞到它们之中海拔最高的一个格子。模拟的过程中,共有两种可能的操作:

\(move\) \(k\)无人机移动k步

\(change\) \(a\) \(b\) \(e\)第\(a\)行第\(b\)列的格子海拔修改为\(e\)。

在每次\(move\)操作后,你都需要立刻找到无人机的位置。你可以认为,每次移动的三个目标位置海拔互不相同,因此每一步移动都是良定义的。

solution

每次移动的步数\(k\)很大,所以可以联想到倍增跳法,即每次跳\(2^k\)步。

但步数不太好维护,我们考虑将移动\(k\)步转化为先暴力跳到第一列,再从第一列出发跳若干圈,最后在暴力跳剩下的不到一圈的步数。显然2段暴力跳的复杂度仅为\(O(c)\),可以接受

那么问题转化为了要维护从第一列的每一个位置出发跳\(2^k\)圈后所处的位置。本题在\(y轴\)上的跳法不确定,但在\(x\)轴上一直是每次向右移动一格,于是可以在列上建立线段树,每个节点(设对应的列为\(l,r\))维护第\(l\)列上每个位置,跳到第\(r+1\)列后所处的位置,就可以

t[p][i]=t[p<<1|1][t[p<<1][i]]

实现转移(其中\(t[p][i]\)表示处在第\(i\)行的点,跳过\(p\)对应的这段区间后所处的位置,在代码中是\(T[p].t[i]\))

于是根节点维护的答案就是第一列中的每个位置跳一圈后所处的位置,接下来的只不过是倍增基本套路。

每次对\((x,y)\)的修改只会影响第\(x-1\)列维护的答案,等于是线段树中的单点修改,复杂度\(O(rlog(c))\)

code

#include<bits/stdc++.h>
using namespace std;
const int N=5010;
int a[N][N],ans[N][N],R,C,m,to[N][32];
inline int read(){
int x=0,f=1;char ch=getchar();
while(!isdigit(ch)){if(ch=='-') f=-1;ch=getchar();}
while(isdigit(ch)){x=(x<<3)+(x<<1)+(ch^48);ch=getchar();}
return x*f;
}
struct SGT{
struct tree{
int t[N];
}T[N<<2];
#define lc (p<<1)
#define rc (p<<1|1)
#define mid (l+r>>1)
inline void pushup(int p){
for(int i=1;i<=R;++i)
T[p].t[i]=T[rc].t[T[lc].t[i]];
}
inline void build(int p,int l,int r){
if(l==r){
for(int i=1;i<=R;++i) T[p].t[i]=ans[l][i];
return ;
}
build(lc,l,mid);
build(rc,mid+1,r);
pushup(p);
}
inline void update(int p,int l,int r,int x){
if(l==r){
for(int i=1;i<=R;++i) T[p].t[i]=ans[l][i];
return ;
}
if(x<=mid) update(lc,l,mid,x);
else update(rc,mid+1,r,x);
pushup(p);
}
#undef lc
#undef rc
#undef mid
}T;
inline void work(int &x,int &y){
int yy=y==C?1:y+1,x1=x>1?x-1:R,x2=x,x3=x==R?1:x+1;
int ans=a[x1][yy],pos=x1;
if(a[x2][yy]>ans) ans=a[x2][yy],pos=x2;
if(a[x3][yy]>ans) ans=a[x3][yy],pos=x3;
x=pos;y=yy;
}
int main(){
R=read();C=read();
for(int i=1;i<=R;++i)
for(int j=1;j<=C;++j)
a[i][j]=read();
for(int i=1;i<=R;++i){
for(int j=1;j<=C;++j){
int x=i,y=j;work(x,y);
ans[j][i]=x;
}
}
T.build(1,1,C);
for(int i=1;i<=R;++i) to[i][0]=T.T[1].t[i];
for(int j=1;j<=30;++j)
for(int i=1;i<=R;++i)
to[i][j]=to[to[i][j-1]][j-1];
m=read();
int nx=1,ny=1;
while(m--){
char s[10];scanf("%s",s);
if(s[0]=='c'){
int x=read(),y=read(),e=read();
a[x][y]=e;y=y>1?y-1:C;
for(int i=1;i<=R;++i){
int t1=i,t2=y;
work(t1,t2);
ans[y][i]=t1;
}
T.update(1,1,C,y);
for(int i=1;i<=R;++i) to[i][0]=T.T[1].t[i];
for(int j=1;j<=30;++j)
for(int i=1;i<=R;++i)
to[i][j]=to[to[i][j-1]][j-1];
}
if(s[0]=='m'){
int k=read();
while(k&&ny!=1) work(nx,ny),k--;
int circle=k/C;k=k%C;
for(int i=30;i>=0;--i) if(circle&(1<<i)) circle^=(1<<i),nx=to[nx][i];
while(k--) work(nx,ny);
printf("%d %d\n",nx,ny);
}
}
return 0;
}

「CERC2017」Donut Drone的更多相关文章

  1. 「译」JUnit 5 系列:条件测试

    原文地址:http://blog.codefx.org/libraries/junit-5-conditions/ 原文日期:08, May, 2016 译文首发:Linesh 的博客:「译」JUni ...

  2. 「译」JUnit 5 系列:扩展模型(Extension Model)

    原文地址:http://blog.codefx.org/design/architecture/junit-5-extension-model/ 原文日期:11, Apr, 2016 译文首发:Lin ...

  3. JavaScript OOP 之「创建对象」

    工厂模式 工厂模式是软件工程领域一种广为人知的设计模式,这种模式抽象了创建具体对象的过程.工厂模式虽然解决了创建多个相似对象的问题,但却没有解决对象识别的问题. function createPers ...

  4. 「C++」理解智能指针

    维基百科上面对于「智能指针」是这样描述的: 智能指针(英语:Smart pointer)是一种抽象的数据类型.在程序设计中,它通常是经由类型模板(class template)来实做,借由模板(tem ...

  5. 「JavaScript」四种跨域方式详解

    超详细并且带 Demo 的 JavaScript 跨域指南来了! 本文基于你了解 JavaScript 的同源策略,并且了解使用跨域跨域的理由. 1. JSONP 首先要介绍的跨域方法必然是 JSON ...

  6. 「2014-5-31」Z-Stack - Modification of Zigbee Device Object for better network access management

    写一份赏心悦目的工程文档,是很困难的事情.若想写得完善,不仅得用对工具(use the right tools),注重文笔,还得投入大把时间,真心是一件难度颇高的事情.但,若是真写好了,也是善莫大焉: ...

  7. 「2014-3-18」multi-pattern string match using aho-corasick

    我是擅(倾)长(向)把一篇文章写成杂文的.毕竟,写博客记录生活点滴,比不得发 paper,要求字斟句酌八股结构到位:风格偏杂文一点,也是没人拒稿的.这么说来,arxiv 就好比是 paper 世界的博 ...

  8. 「2014-3-17」C pointer again …

    记录一个比较基础的东东-- C 语言的指针,一直让人又爱又恨,爱它的人觉得它既灵活又强大,恨它的人觉得它太过于灵活太过于强大以至于容易将人绕晕.最早接触 C 语言,还是在刚进入大学的时候,算起来有好些 ...

  9. 「2014-3-13」Javascript Engine, Java VM, Python interpreter, PyPy – a glance

    提要: url anchor (ajax) => javascript engine (1~4 articles) => java VM vs. python interpreter =& ...

随机推荐

  1. apktool重新打包添加签名

    一.生成apk apktool b 反编译后项目目录 -o 新apk名称.apk 二.生成签名 keytool -genkeypair -alias 新apk名称.apk -keyalg RSA -v ...

  2. libhiredis.so.0.13: cannot open shared object file: No such file or directory in Unknown on line

    vim /etc/ld.so.conf添加 /usr/local/lib (此处为动态链接库的目录) ldconfig

  3. 多线程之Callable

    多线程实现Callable的好处有三点 1.Callable支持泛型 2.Callable支持返回值 3.Callable可以抛出异常 class MyThread2 implements Calla ...

  4. 【转】Setting up SDL on Windows

    FROM: http://lazyfoo.net/tutorials/SDL/01_hello_SDL/windows/index.php Setting up SDL on Windows Last ...

  5. DOM4J API

    1.DOM4J简介 DOM4J是 dom4j.org 出品的一个开源 XML 解析包.DOM4J应用于 Java 平台,采用了 Java 集合框架并完全支持 DOM,SAX 和 JAXP. DOM4J ...

  6. SQL Server 列存储索引 第四篇:实时运营数据分析

    实时运营数据分析(real-time operational analytics )是指同时在同一张数据表上执行分析处理和业务处理.分析查询主要是对海量数据执行聚合查询,而事务主要是指对数据表进行少量 ...

  7. active cab inf文件编写

    最近做了一个网页下载控件.主要就是实现ActiveX控件功能. 由于自己是第一次做,不熟悉其过程.中间走了很多弯路.现在把走过得路程记录部分,希望对其他人可以有点用. 首先制作一个你自己的DLL文件. ...

  8. Nginx是什么?有什么用?

    一.Nginx是什么 Nginx ("engine x") 是一个高性能的 HTTP 和反向代理服务器,特点是占有内存少,并发能力强,事实上nginx的并发能力确实在同类型的网页服 ...

  9. 如何对List集合中的对象进行按某个属性排序

    我们在实际的开发工作中,经常会碰到排序的问题,如题,我们如何针对List集合中的某一个属性进行排序 当list集合中的元素类型是我们自定义类型时,有两种对list中的元素进行排序的方法: 方法一 让l ...

  10. Docker(12)- docker run 命令详解

    如果你还想从头学起 Docker,可以看看这个系列的文章哦! https://www.cnblogs.com/poloyy/category/1870863.html 作用 创建一个新的容器并运行一个 ...