传送门

网上大部分题解都写得是动态开点线段树,然而像\(MiEcoku\)这么懒惰的显然不会去写线段树...

\(\color{green}{solution}\)

我们考虑来点骚操作. 线段树维护的是区间最长连续空段,那么我们能不能单独把空的区间提出来呢?这样就可以
不用线段树了.

对于连续的一段空区间,我们可以用一个\(set\)去维护. 那么空区间的问题解决了,现在怎么求区间内有多少个插排被使用呢?
这...如果没有同学离开,那么随便用一个数据结构就可以维护. 但因为我 们需要支持删除操作,所以我用平衡树维护.

具体的删除操作:
在一位同学离开后,假设他的位置是\(p\), 他前面第一个被使用的位置是\(p_l\), 他后面第一个
被使用的位置是\(p_r\), 那么我们只需要再\(set\)中删除区间 \([p_l+1, p-1]\) 和区间 \([p+1, p_r-1]\)(如果区间存在)
再将区间 \([p_l+1, p_r-1]\) 加入\(set\)即可

具体实现详见代码

#include <bits/stdc++.h>
using namespace std;

const int maxn = 100010;

struct line {
    int l, r;
    bool operator < (const line &a) const {
        return r-l == a.r-a.l ? r > a.r : r-l > a.r-a.l;
    }
    line (int l, int r) : l(l), r(r) {}
    line () {}
};
set<line> tc; map<int, int> id;
struct node {int v, ky, sz; node *ls, *rs;};
node *nul, pool[maxn], *pis = pool, *rt;
inline void init() {
    nul=pis; nul->ls=nul->rs=nul; nul->sz=0; rt=nul; srand(19260817);
}
node *newnode(int val) {
    node *k = ++ pis; k->ls = k->rs = nul;
    k->v=val; k->sz = 1; k->ky = rand(); return k;
}
inline void up(node *&x) { x->sz = x->ls->sz + x->rs->sz + 1;}
inline void spl(node *c, node *&x, node *&y, int val) {
    if( c == nul) { x = y = nul; return;}
    if( c->v <= val) x = c, spl(c->rs, x->rs, y, val);
    else y = c, spl(c->ls, x, y->ls, val); up(c);
}
inline void mrg(node *&c, node *x, node *y) {
    if( x==nul || y==nul) {c = x==nul ? y : x; return;}
    if( x->ky < y->ky) c = x, mrg(c->rs, x->rs, y);
    else c = y, mrg(c->ls, x, y->ls); up(c);
}
int insert() {
    line c = *tc.begin(); tc.erase(tc.begin());
    int pos = c.r + c.l + 1 >> 1;
    tc.insert(line(c.l,pos-1)); tc.insert(line(pos+1,c.r));
//  printf("%d -\n", pos); // de bug
    node *x, *y;
    spl(rt, x, y, pos); mrg(x, x, newnode(pos)); mrg(rt, x, y);
    return pos;
}
inline void erase(int opt) {
    node *x, *y, *z, *c; spl(rt, x, y, opt); spl(x, x, z, opt-1);
    int l, r;
    c=x; while( c->rs!=nul) c = c->rs; l = c->v;
    c=y; while( c->ls!=nul) c = c->ls; r = c->v;
    mrg(rt, x, y);
    tc.erase(line(l+1, opt-1)); tc.erase(line(opt+1, r-1));
    tc.insert(line(l+1, r-1));
}
inline void modify(int opt) {
    if( id[opt]) erase(id[opt]), id[opt] = 0;
    else id[opt] = insert();
}
inline void qry(int l, int r) {
    node *x, *y, *z; spl(rt, x, y, r); spl(x, x, z, l-1);
    printf("%d\n", z->sz); mrg(x, x, z); mrg(rt, x, y);
}
int n, Q;
int main() {
#ifndef ONLINE_JUDGE
    freopen("1.in","r",stdin);
    freopen("b.out","w",stdout);
#endif
    init();
    scanf("%d%d", &n, &Q); tc.insert(line(1, n));
    mrg(rt, rt, newnode(0)); mrg(rt, rt, newnode(n+1));
    for ( register int i = 1, opt, l, r; i <= Q; ++ i) {
        scanf("%d", &opt);
        if( opt) modify(opt);
        else scanf("%d%d", &l, &r), qry(l, r);
    }
    return 0;
}

[BZOJ 5155][Tjoi2014]电源插排的更多相关文章

  1. P3968 [TJOI2014]电源插排

    P3968 [TJOI2014]电源插排 线段树维护最长空区间及左端点位置,这个和$nlongn$的动态最大子序和差不多,就不多解释了 $n$较大哈希优化空间 My complete code: #i ...

  2. vijos1859[TJOI2014]电源插排

    题意:小 M 的实验室有很多电源插排.这些插排的编号从 1 到 N,由左向右排成一排.每天早晨,这些插排都是没有被使用的.每当一个学生来到实验室,他就将自己的笔记本电源插到某一个未被使用的插排上.实验 ...

  3. [BZOJ 5158][Tjoi2014]Alice and Bob

    传送门 \(\color{green}{solution}\) 贪心 /************************************************************** P ...

  4. 【刷题】BZOJ 5154 [Tjoi2014]匹配

    Description 有N个单身的男孩和N个单身女孩,男孩i和女孩j在一起得到的幸福值为Hij.一个匹配即对这N个男孩女孩的安排: 每个男孩恰好有一个女朋友,每个女孩恰好有一个男朋友.一个匹配的幸福 ...

  5. BZOJ 2127: happiness [最小割]

    2127: happiness Time Limit: 51 Sec  Memory Limit: 259 MBSubmit: 1815  Solved: 878[Submit][Status][Di ...

  6. BZOJ 3275: Number

    3275: Number Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 874  Solved: 371[Submit][Status][Discus ...

  7. BZOJ 2879: [Noi2012]美食节

    2879: [Noi2012]美食节 Time Limit: 10 Sec  Memory Limit: 512 MBSubmit: 1834  Solved: 969[Submit][Status] ...

  8. bzoj 4610 Ceiling Functi

    bzoj 4610 Ceiling Functi Description bzoj上的描述有问题 给出\(n\)个长度为\(k\)的数列,将每个数列构成一个二叉搜索树,问有多少颗形态不同的树. Inp ...

  9. BZOJ 题目整理

    bzoj 500题纪念 总结一发题目吧,挑几道题整理一下,(方便拖板子) 1039:每条线段与前一条线段之间的长度的比例和夹角不会因平移.旋转.放缩而改变,所以将每条轨迹改为比例和夹角的序列,复制一份 ...

随机推荐

  1. [Selenium] 在Grid模式下打印出当前Case是在哪台Node上运行

    AAAbstractFlow() public void getComputerNameOfNode(WebDriver driver){ String CIHub = Environment.get ...

  2. HBase 系列(二)安装部署

    HBase 系列(二)安装部署 本节以 Hadoop-2.7.6,HBase-1.4.5 为例安装 HBase 环境.HBase 也有三种模式:本地模式.伪分布模式.分布模式. 一.环境准备 (1) ...

  3. POP邮件收取邮件 代码

    // 111111.cpp : 定义控制台应用程序的入口点. // #include "stdafx.h" #include <WinSock.h> #include ...

  4. CSS 如何让Table的里面TD全有边框 而Table的右左边框没有

    比如这样一个CSS.td3{font-size: 14px;color: #FFFFFF;background-color: #000000;BORDER-RIGHT: #f6f6f6 1px sol ...

  5. 安装wampserver后,在www文件夹下面写php文件,而在网页里输入localhost而无法打开php文件时解决办法汇总

    wampserver安装后,在www文件夹下面写入xx.PHP文件,然后在网页里输入localhost:xx.PHP. 你可能会遇到如下三种情况: 情形一:网页上显示空白,按F12,出现404的错误. ...

  6. Gym 101201H Paint (离散化+DP)

    题意:给定 n 个区间,让你选出一些,使得每个选出区间不交叉,并且覆盖区间最大. 析:最容易想到的先是离散化,然后最先想到的就是 O(n^2)的复杂度,dp[i] = max(dp[j] + a[i] ...

  7. Gym 101201F Illumination (Two-Sat)

    题意:一个n*n的房子,有很多灯,每个格子只能被上下方向照一次.左右方向照一次,每个灯可以选择上下或是左右照,照明长度以自身位置为中心,占用2*r+1个格子.问能否安排一种方案,使所有格子满足条件. ...

  8. 从《数据挖掘概念与技术》到《Web数据挖掘》

    从<数据挖掘概念与技术>到<Web数据挖掘> 认真读过<数据挖掘概念与技术>的第一章后,对数据挖掘有了更加深刻的了解.数据挖掘是知识发展过程的一个步骤.知识发展的过 ...

  9. 看图说说Heap中对象的生存状态

  10. Web 应用简单测试方案

    测试:一定要分阶段测试,先确定入队列成功,再测试队列的执行是否成功. 功能点: 1. 翻页2. 加精3. 置顶4. 帖子浏览量(PV)5. 发帖6. 回复7. 评论 8. crontab 脚本 @20 ...