[Luogu P2827] 蚯蚓 (巧妙的模拟)
题面:
传送门:https://www.luogu.org/problemnew/show/P2827
Solution
看到这题,我们肯定会有一个大胆想法。
那就是直接用堆模拟这个过程。
对于q,我们只需要在堆中多维护一个T,记录每个点插入的时间,在新的元素插入时直接计算所比较的点的当前长度就可以完成插入了。
时间复杂度O(M*log(M))
这样的做法只能获得65-70分,因为后面的数据非常大。
所以说,我们要另寻他路。
首先,我们经过看题解手玩可以发现一个很显然但是很重要的结论:
在q=0的时候,一条线段所分裂出来的两条线段肯定要比它更小的线段分裂出来的对应的两条线段更大。
证明十分简单,设x,y为两条分裂前的线段,且x>y
那么,较长的那一条(假设p>0.5(p为分割点)) 为: px 与 py, 显然,px>py
同理可证另一条线段也有这种关系。
根据这个关系,我们可以考虑这样的做法:
我们开三个队列,第一个队列放入排好序的原序列,第二个队列放每次分裂出来的较长的蚯蚓,第三个放每次分裂出来的较短的蚯蚓。
那么根据刚才的证明,我们可以得出,第二个与第三个队列一定是有序的,因为我们每次取的蚯蚓一定比之前取的更短,所以分裂出来的肯定比比之前分裂出来的蚯蚓更短。
这样子,我们就可以模拟这个过程,每次取三个队列中最大的那一个,并把分裂出来的对应放到第二第三个队列的末尾就好。
事实上,对于q>0的情况,这个推论也是成立的。
首先,我们可以假设出来x与y分别是分裂前的线段长度且x>y,假设x与y之间间隔的时间为T
那么,在y分裂的时刻,x分裂出来的线段较长的长度为(假设p>0.5):px+T*q ; y分裂出来的较长的线段长度为: (y+T*q)*p = py + T*q*p
显然 px+T*q > py + T*q*p
所以说,我们刚刚的结论在这里也是成立的。
对于在某一时刻的线段的具体长度,我们可以通过在队列中多记录一个插入时间,这样就可以算出某一时刻的某条线段的具体长度了。
时间复杂度O(nlogn+m)
就酱,我们就可以AC这道题啦(≧ω≦)/
Code
//Luogu P2827 蚯蚓
//Sep,9th,2018
//巧妙的三个队列
#include<iostream>
#include<cstdio>
#include<queue>
#include<algorithm>
using namespace std;
long long read()
{
long long x=0,f=1; char c=getchar();
while(!isdigit(c)){if(c=='-') f=-1;c=getchar();}
while(isdigit(c)){x=x*10+c-'0';c=getchar();}
return x*f;
}
const int N=100000+100;
const int M=7000000+N+2000;
int a[M],n,m,q,u,v,t;
struct worm
{
int T,s;
worm (int A,int B)
{
s=A,T=B;
}
inline int GetLen(int mtime)//mtime秒后的长度
{
return s+(mtime-T)*q;
}
};
queue <worm> A,B,C;
double p;
bool cmp(int x,int y)
{
return x>y;
}
int main()
{
n=read(),m=read(),q=read(),u=read(),v=read(),t=read();
for(int i=1;i<=n;i++)
a[i]=read(); sort(a+1,a+1+n);
for(int i=n;i>=1;i--)
A.push(worm(a[i],0));
for(int i=1;i<=m;i++)
{
long long from=-1,t_MAX=-0x3f3f3f3f;
if(A.empty()==false) from=1,t_MAX=A.front().GetLen(i-1);
if(B.empty()==false and B.front().GetLen(i-1)>t_MAX) from=2,t_MAX=B.front().GetLen(i-1);
if(C.empty()==false and C.front().GetLen(i-1)>t_MAX) from=3,t_MAX=C.front().GetLen(i-1);
int px=(t_MAX*u)/v,t_px=t_MAX-px;
B.push(worm(max(px,t_px),i));
C.push(worm(min(px,t_px),i));
if(from==1) A.pop();
else if(from==2) B.pop();
else C.pop();
if(i%t==0)
printf("%lld ",t_MAX);
}
printf("\n"); n=0;
while(A.empty()==false)
a[++n]=A.front().GetLen(m),A.pop();
while(B.empty()==false)
a[++n]=B.front().GetLen(m),B.pop();
while(C.empty()==false)
a[++n]=C.front().GetLen(m),C.pop();
sort(a+1,a+1+n,cmp);
for(int i=1;i<=n;i++)
if(i%t==0)
printf("%d ",a[i]);
return 0;
}
[Luogu P2827] 蚯蚓 (巧妙的模拟)的更多相关文章
- Luogu P2827 蚯蚓(模拟)
P2827 蚯蚓 题意 题目描述 本题中,我们将用符号\(\lfloor c\rfloor\)表示对\(c\)向下取整,例如:\(\lfloor 3.0\rfloor =\lfloor 3.1\rfl ...
- Luogu P2827 蚯蚓
看到题目就可以想到直接开的堆模拟的过程了吧,这个还是很naive的 注意在用堆做的时候也是要明智一点的,对于蚯蚓长度的相加肯定不能直接遍历并加上,还是可以差分一下的 其实说白了就是把集体加->单 ...
- 【luogu P2827 蚯蚓】 题解
题目链接:https://www.luogu.org/problemnew/show/P2827 35分:暴力sortO(mnlogn). 80分:考虑到每次不好维护不被切的点+q,正难则反.改成维护 ...
- LUOGU P2827 蚯蚓 (noip 2016)
传送门 解题思路 第一眼以为是一个二叉堆,直接上优先队列60分...后来听ztz11说有单调性,新加入的蚯蚓一定比原先在的蚯蚓长度长,开三个队列,分别放原先的长度,切掉后大的那一半,切掉后小的那一半. ...
- 洛谷P2827 蚯蚓——思路题
题目:https://www.luogu.org/problemnew/show/P2827 思路... 用优先队列模拟做的话,时间主要消耗在每次的排序上: 能不能不要每次排序呢? 关注先后被砍的两条 ...
- 洛谷——P2827 蚯蚓
P2827 蚯蚓 题目描述 本题中,我们将用符号 \lfloor c \rfloor⌊c⌋ 表示对 cc 向下取整,例如:\lfloor 3.0 \rfloor = \lfloor 3.1 \rflo ...
- 【巧妙的模拟】【UVA 10881】 - Piotr's Ants/Piotr的蚂蚁
</pre></center><center style="font-family: Simsun;font-size:14px;"><s ...
- 洛谷P2827 蚯蚓 题解
洛谷P2827 蚯蚓 题解 题目描述 本题中,我们将用符号 ⌊c⌋ 表示对 c 向下取整. 蛐蛐国最近蚯蚓成灾了!隔壁跳蚤国的跳蚤也拿蚯蚓们没办法,蛐蛐国王只好去请神刀手来帮他们消灭蚯蚓. 蛐蛐国里现 ...
- 【BZOJ】4721: [Noip2016]蚯蚓 / 【洛谷】P2827 蚯蚓(单调队列)
Description 本题中,我们将用符号[c]表示对c向下取整,例如:[3.0」= [3.1」=[3.9」=3.蛐蛐国最近蚯蚓成灾了!隔壁跳 蚤国的跳蚤也拿蚯蚓们没办法,蛐蛐国王只好去请神刀手来帮 ...
随机推荐
- Windows10下JDK8的下载安装与环境变量的配置
Windows10下JDK8的下载安装与环境变量的配置 下载JDK8(64位) 链接:https://pan.baidu.com/s/10ZMK7NB68kPORZsPOhivog 提取码:agsa ...
- Mysql中 int(3) 类型的含义
注意:这里的(3)代表的并不是存储在数据库中的具体的长度,以前总是会误以为int(3)只能存储3个长度的数字,int(11)就会存储11个长度的数字,这是大错特错的. 其实当我们在选择使用int的类型 ...
- 探究"补阶乘大法的本质"——糖水不等式!
废话不多说先来康一条例题: 证明: 下面给出题目的一种解法(我称之为"补阶乘大法"): 思考:为什么补上一个阶乘(准确说不是阶乘,是两个数阶乘的之商)项,放缩后再给去掉,就能达到我 ...
- const放在函数前后的区别
转载:const放在函数前后的区别 一.const修饰指针 int b = 500; 1.const int * a = & b; 2.int const * a = & b; 3.i ...
- JS原生练习
1.输出1-10000之间的数 <script> for(i=1;i<=10000;i++) { document.write(i + "<br>") ...
- Makefile-3-书写规则
目录 前言 概念 Chapter 3:书写规则 3.3 在规则中使用通配符 3.4 文件搜索 3.8 自动生成依赖性 原理 * 直接解析例子 ** sed 命令 参考 书籍 前言 本笔记主要记录Mak ...
- 2014年 实验五 Internet与网络工具的使用
实验五 Internet与网络工具的使用 [实验目的] ⑴.FTP服务器的架设和客户端的使用. ⑵.使用云盘和云笔记应用 ⑶.运用QQ的远程协助功能. (4).默认安装foxmail软件,进行邮件 ...
- MeteoInfoLab脚本示例:MERRA HDF数据
MERRA是NOAA的一种再分析资料,HDF数据遵循COARDS协议,读取比较简单.脚本程序: #Add data file folder = 'D:/Temp/hdf/' fns = 'MERRA3 ...
- day57 Pyhton 前端Jquery09
内容回顾: - 筛选选择器 $('li:eq(1)') 查找匹配的元素 $('li:first') $('li:last') - 属性选择器 - 筛选的方法 - find() 查找后代的元素 - ...
- json对象去重,根据指定字段
function FilterByName(data, Name) { //data是json对象,Name是根据什么字段去重 var map = {}, dest = []; for (var i ...