[NOIp2016] 蚯蚓

类型:单调队列

传送门:>Here<

题意:有$N$只蚯蚓,每秒都会伸长$q$。每一次都会有人选出最长的一条切成两半,长度分别是$\left \lfloor px \right \rfloor$和$x - \left \lfloor px \right \rfloor$ 询问每一秒最长的蚯蚓被切前的长度,以及$m$秒后每条蚯蚓的长度(从大到小排序)

解题思路

NOIp的subtask还是非常良心的。于是决定不看题解开始干……

看完题目,花了10分钟打了一个超级暴力模拟,35分get 然后发现有60分是$q=0$的情况……这不是就是一个裸的堆吗?打了15分钟,50分get(为什么只有50……) 然后想了很久没思路,瞟了一眼题解一眼就看到去减长度而不用管加,赶紧开始打调试近40分钟后85分get 然后就开始看题解打正解了……正解也调了20分钟左右 最终的AC代码再交一次竟然变成了90 又交一次变成了95 再交一次又100了……洛谷的评测机也不是很稳定啊,这题还是有点卡常

先来看看85分怎么拿。85分的写法在思想上还是很重要的——蚯蚓每秒增长$q$,可以看做是被切的蚯蚓减去$q$。因此我们可以维护一个堆,这样堆的内部就不需要反复更新。但是细节要注意,选择切割的蚯蚓长度应该拿真实的长度来算

然后来看正解。很容易发现,对于两条蚯蚓$x,y$,如果$x>y$,则$x$肯定会先被切掉。不妨设$x$被切掉以后变为$\{a_1, b_1\}$,$y$被切后变为$\{a_2, b_2\}$

由于$y$肯定在$x$被切后若干秒被切,不妨设为$t$秒,则那时四条分出来的小蚯蚓的长度是可以表示的。我们希望能够证明到那时$a_1 > a_2, b_1 > b_2$。我们先来证明$a_1>a_2$,$b$也类似$$a_1 = a_1 + q*t =\left \lfloor px \right \rfloor + q*t$$$$a_2 = \left \lfloor p(y+q*t) \right \rfloor$$则$$a_1-a_2=\left \lfloor px \right \rfloor + q*t-\left \lfloor p(y+q*t) \right \rfloor$$去掉向下取整符号并不会影响答案,因此$$a_1-a_2=p*x+q*t-p*y-p*q*t$$整理得$$a_1-a_2=p(x-y)+q*t(1-p)>0$$故$$a_1>a_2$$

因此我们得出结论:一条蚯蚓如果比另一条蚯蚓早被切,那么它分出来的两条蚯蚓也永远比后分出来的两条蚯蚓要长

于是我们可以考虑维护三个队列$q_1,q_2,q_3$,$q_1$中储存没被切过的蚯蚓,$q_2$中储存切出来的左半条,$q_3$表示右半条。要求队列单调不上升,每一次取出三个队列的队头中的最大值即为所有蚯蚓中最长的,切成两半后分别塞到$q_2,q_3$的队尾。由于刚才证明了后切的一定要短,所以插入队尾后单调性依然满足。故这样的做法是正确的

Code

cur=-INF而不能是-1,因为负数有可能减到很大

/*By DennyQi 2018.8.15*/
#include <cstdio>
#include <queue>
#include <cstring>
#include <algorithm>
#define  r  read()
#define  Max(a,b)  (((a)>(b)) ? (a) : (b))
#define  Min(a,b)  (((a)<(b)) ? (a) : (b))
using namespace std;
typedef long long ll;
const int MAXN = 7000010;
const int MAXM = 27010;
const int INF = 1061109567;
inline int read(){
    int x = 0; int w = 1; register int c = getchar();
    while(c ^ '-' && (c < '0' || c > '9')) c = getchar();
    if(c == '-') w = -1, c = getchar();
    while(c >= '0' && c <= '9') x = (x << 3) + (x << 1) + c - '0', c = getchar(); return x * w;
}
double p;
int N,M,Q,U,V,T,_mx,pos,tmp,x,y,cur,top;
int q[4][MAXN],h[4],t[4],a[MAXN];
inline bool comp(const int& a, const int& b){ return a>b; }
int main(){
    N=r,M=r,Q=r,U=r,V=r,T=r;
    p = (double)(U) / (double)(V);
    for(int i = 1; i <= N; ++i) a[i] = r;
    h[1] = h[2] = h[3] = 1;
    q[2][1] = q[3][1] = -INF;
    sort(a+1,a+N+1,comp);
    for(int i = 1; i <= N; ++i) q[1][++t[1]] = a[i];
    for(int _t = 1; _t <= M; ++_t){
        cur = -INF;
        for(int i = 1; i <= 3; ++i){
            if(h[i] > t[i]) continue;
            if(q[i][h[i]] > cur){ cur = q[i][h[i]]; pos = i; }    
        }
        if(_t % T == 0) printf("%d ", cur+(_t-1)*Q);
        ++h[pos];
        x = (cur+(_t-1)*Q) * p, y = (cur+(_t-1)*Q) - x;
        q[2][++t[2]] = x-_t*Q, q[3][++t[3]] = y-_t*Q;
    }
    puts("");
    for(int i = 1; i <= 3; ++i)
        for(int j = h[i]; j <= t[i]; ++j) a[++top] = q[i][j];
    sort(a+1,a+top+1,comp);
    for(int i = 1; i <= top; ++i)
        if(i % T == 0) printf("%d ", a[i]+M*Q);
    return 0;
}
posted @ 2018-08-15 15:45  DennyQi  阅读(186)  评论(0编辑  收藏  举报