【ARC082D】Sandglass
Description
题目链接
Description
好题。题意是维护一个初始值,交替加减一段时间,有上界\(m\)和下界0(不能超过这两条界限),问对于某一种初始值,在某一个时刻时该值为多少?
可以把所有询问按时间排序成一列,然后用线段树区间加减、区间min、max暴力实现,然而我不会做。
实际上直接模拟即可。
如果按时间增长为横坐标,按该时间的权值为纵坐标,画出对于\([0,m]\)的每一种初始值的函数图像,我们会发现,随着时间的增长,函数曲线变得原来越少,也就是图像会不断重叠在一起。
形式化地讲,我们可以发现一个重要性质:对于一个在某一时刻触碰到上/下边界的曲线,在这个时刻之后,其函数图像将会和在这个时刻之前已触碰到上/下边界的曲线完全相同。
在某一时刻时,我们称一个初始值为“归上”,当且仅当其在这个时刻及以前已经触碰到了上边界;“归下“同理。我们发现,对于任意确定时刻,”归上“的初始值都是\([y,m]\),而"归下"的初始值都是\([0,x]\)。而且随着时间推进,\(y\)单调不增,\(x\)单调不减。
既然询问已经以时间递增的顺序给出,那我们就顺序模拟时间推进,并逐一处理询问。
我们要维护的东西有三个:"归上"初始值在当前时间的具体取值\(up\)、“归下”的初始值在当前时间的具体取值\(dn\),以及从开始到现在上下移动的总和\(sum\)。
对于一个询问\((t,a)\),假设时间已经模拟到了\(t\),如果\(a\)"归上",也就是曾经碰到上边界,那么答案就是\(up\);如果其"归下",则答案就是\(dn\);否则,\(a\)从开始到现在没有碰到任何边界,所以其取值直接模拟即可,恰好为\(a+sum\)。至于”是否曾经触边“的判定,可以维护\(sum\)有史以来的最大值和最小值,加在\(a\)上判定与0或m的大小关系即可。
对于\(up\)和\(dn\)的计算,在初始时令\(dn=0\),\(up=m\),然后在时间推进的过程中不断对它们进行带边界限制的模拟上下移动,那么我们就可以保证在每一个时刻时,\(up\)和\(dn\)都是我们所定义的值。为什么?因为初始时,“归上”恰好只有\(m\),"归下"恰好只有0。而之后触碰到上/下边界的所有曲线,都必定在m/0之后触碰,也必定在m/0之后成为"归上"/“归下"。只要一触碰,其函数值就会和m/0相同。于是本质上,我们是在维护\(a=0\)和\(a=m\)在任意时刻的取值。
形象地讲,这道题就像一个非弹性形变的柱子在管道里上下移动,柱子的”长度“就代表着那一部分从未触边的初始值,而上面和下面空出来的部分,代表着上面这些取值的答案都是柱子的”上端“,下面这些初始值的答案都是柱子的“下端”。为什么会空出来呢?因为曾经被“挤”在一起了,因此这一部分初始值在以后的取值都相同了。那么\(up\)和\(dn\)其实就是柱子的上下端。
Code
#include <cstdio>
using namespace std;
const int N=100005;
int n,m,q,a[N];
int l,r,sum,summn,summx;
inline int min(int x,int y){
return x<y?x:y;
}
inline int max(int x,int y){
return x>y?x:y;
}
inline void move(int &x,int d,int tim){
x+=d*tim;
if(x>m) x=m;
if(x<0) x=0;
}
void cont(int tim,int d){
static int lasttim=0;
int delta=tim-lasttim;
lasttim=tim;
sum+=delta*d;
summn=min(summn,sum);
summx=max(summx,sum);
move(l,d,delta);
move(r,d,delta);
}
int query(int x){
if(x+summn<=0) return l;
else if(x+summx>=m) return r;
else return x+sum;
}
int main(){
scanf("%d%d",&m,&n);
for(int i=1;i<=n;i++)
scanf("%d",a+i);
scanf("%d",&q);
l=0; r=m;
sum=summx=summn=0;
int i=1,d=-1,x,y;
while(q--){
scanf("%d%d",&x,&y);
for(;i<=n&&a[i]<=x;cont(a[i++],d),d=-d);
cont(x,d);
printf("%d\n",query(y));
}
return 0;
}
【ARC082D】Sandglass的更多相关文章
- 【AtCoder】ARC082 F - Sandglass
[链接]F - Sandglass [题意]给定沙漏A和B,分别装着a和X-a的沙子,开始时A在上B在下,每秒漏1,漏完不再漏.给定n,有n个时刻ai沙漏倒转.给定m个询问,每次询问给定初值a和时刻t ...
- 【AtCoder Regular Contest 082 F】Sandglass
[链接]点击打开链接 [题意] 你有一个沙漏. 沙漏里面总共有X单位的沙子. 沙漏分A,B上下两个部分. 沙漏从上半部分漏沙子到下半部分. 每个时间单位漏1单位的沙子. 一开始A部分在上面.然后在r1 ...
- Python高手之路【六】python基础之字符串格式化
Python的字符串格式化有两种方式: 百分号方式.format方式 百分号的方式相对来说比较老,而format方式则是比较先进的方式,企图替换古老的方式,目前两者并存.[PEP-3101] This ...
- 【原】谈谈对Objective-C中代理模式的误解
[原]谈谈对Objective-C中代理模式的误解 本文转载请注明出处 —— polobymulberry-博客园 1. 前言 这篇文章主要是对代理模式和委托模式进行了对比,个人认为Objective ...
- 【原】FMDB源码阅读(三)
[原]FMDB源码阅读(三) 本文转载请注明出处 —— polobymulberry-博客园 1. 前言 FMDB比较优秀的地方就在于对多线程的处理.所以这一篇主要是研究FMDB的多线程处理的实现.而 ...
- 【原】Android热更新开源项目Tinker源码解析系列之一:Dex热更新
[原]Android热更新开源项目Tinker源码解析系列之一:Dex热更新 Tinker是微信的第一个开源项目,主要用于安卓应用bug的热修复和功能的迭代. Tinker github地址:http ...
- 【调侃】IOC前世今生
前些天,参与了公司内部小组的一次技术交流,主要是针对<IOC与AOP>,本着学而时习之的态度及积极分享的精神,我就结合一个小故事来初浅地剖析一下我眼中的“IOC前世今生”,以方便初学者能更 ...
- Python高手之路【三】python基础之函数
基本数据类型补充: set 是一个无序且不重复的元素集合 class set(object): """ set() -> new empty set object ...
- Python高手之路【一】初识python
Python简介 1:Python的创始人 Python (英国发音:/ˈpaɪθən/ 美国发音:/ˈpaɪθɑːn/), 是一种解释型.面向对象.动态数据类型的高级程序设计语言,由荷兰人Guido ...
随机推荐
- MapReduce -- 最短路径
示例: 给出各个节点到相邻节点的距离,要求算出初始节点到各个节点的最短路径. 数据: A (B,) (D,) B (C,) (D,) C (E,) D (B,) (C,) (E,) E (A,) (C ...
- 大数据入门第二十三天——SparkSQL(一)入门与使用
一.概述 1.什么是sparkSQL 根据官网的解释: Spark SQL is a Spark module for structured data processing. 也就是说,sparkSQ ...
- Luogu P1273 有线电视网
最近写DP写得比较多了 但是POJ上的题目太傻比了而且不想看英文的题面,然后就在Luogu的试炼场里找了一个DP EX专题写了一下(大概3days吧,一天一题差不多) 这是一道比较简单的DP 话说树形 ...
- MySQL清理慢查询日志slow_log的方法
一.清除原因 因为之前打开了慢查询,导致此表越来越大达到47G,导致磁盘快被占满,使用xtrabackup进行备份的时候文件也超大. mysql> show variables like 'lo ...
- mybatis 初步使用(IDEA的Maven项目, 超详细)
目录 创建 Maven 项目 Maven配置 pom.xml 创建数据库 配置Mybatis 配置mybatis的XML文件 创建实体类和对应的Mapper.xml 测试 源码 @ 创建 Maven ...
- JVM技术周报第2期
JVM技术周报第2期 JVM技术周报分享JVM技术交流群的讨论内容,由群内成员整理归纳而成.如果你有兴趣入群讨论,请关注「Java技术精选」公众号,通过右下角菜单「入群交流」加我好友,获取入群详情. ...
- MyBatis的一级缓存和二级缓存简介笔记
关于mybatis中一级缓存和二级缓存的简单介绍 mybatis的一级缓存: MyBatis会在表示会话的SqlSession对象中建立一个简单的缓存,将每次查询到的结果结果缓存起来,当下次查询的时候 ...
- sudo apt-get update 去除设置的代理
今天想装个软件(wine),使用 sudo apt-get update 命令时,发现给出很多Ign 语句,总出现 Connecting to proxy.http://10.0.126.1:1312 ...
- Asp.Net_优化
ASP.NET: 一.返回多个数据集 检查你的访问数据库的代码,看是否存在着要返回多次的请求.每次往返降低了你的应用程序的每秒能够响应请求的次数.通过在单个数据库请求中返回多个结果集,可以减少与数据库 ...
- B1022. D进制的A+B
除基取余法 #include<bits/stdc++.h> using namespace std; stack<int> s; int main(){ long long a ...