题目

有一个 \(n\times m\) 的方阵,每次出来一个人后向左看齐,向前看齐,询问每次出来的人的编号。

\(n\le 3\times 10^5\)

分析

我们考虑离队本质上只有两种操作:

  • 删除
  • 放入末尾

发现这显然可以用平衡树处理,这里我选用Splay。

需要注意的是,空间不可能开到 \(9\times 10^{10}\),只能动态开点,发现队列总是有大部分是连续的编号,所以每个节点储存一个标号范围,若需要访问到中间的标号 \(k\),将该节点分裂成三个节点\([l,k - 1],\ [k, k],\ [k + 1, r]\)即可。

时间复杂度 \(\Theta(n\log n)\),空间复杂度 \(\Theta(n\log n)\)。可以通过。

代码

#include <bits/stdc++.h>

using namespace std;

typedef long long ll;

const int MAXN = 3e5 + 5;

ll n, m;

struct node {
ll left, right, size;
node *child[2], *father; void updata() {size = child[0]->size + child[1]->size + right - left + 1;}
} *nul = new node; void init() {
nul->child[0] = nul->child[1] = nul->father = nul;
nul->left = 1; nul->right = nul->size = 0;
} node *newNode(ll l, ll r, node *fa) {
node *ptr = new node;
ptr->father = fa; ptr->child[0] = nul; ptr->child[1] = nul;
ptr->left = l; ptr->right = r; ptr->size = r - l + 1;
return ptr;
} void cect(node *x, node *y, ll p) {x->father = y; y->child[p] = x;}
bool getPos(node *x) {return x->father->child[1] == x;} void rotate(node *x) {
ll p = getPos(x), fp = getPos(x->father);
node *gfa = x->father->father;
cect(x->child[p ^ 1], x->father, p);
cect(x->father, x, p ^ 1); cect(x, gfa, fp);
x->child[p ^ 1]->updata(); x->updata();
} void split(node *x, ll k) {
node *tmpChild[2] = {x->child[0], x->child[1]};
if(x->left < k) {
x->child[0] = newNode(x->left, k - 1, x);
if(tmpChild[0] != nul) cect(tmpChild[0], x->child[0], 0);
x->child[0]->updata();
}
if(x->right > k) {
x->child[1] = newNode(k + 1, x->right, x);
if(tmpChild[1] != nul) cect(tmpChild[1], x->child[1], 1);
x->child[1]->updata();
}
x->left = x->right = k;
} struct splayTree {
node *rt; splayTree() {rt = newNode(1, 0, nul);} void build(node *&cur, node *fa, ll l, ll r) {
if(l <= r) {
ll mid = (l + r) >> 1;
cur = newNode(mid * m, mid * m, fa);
build(cur->child[0], cur, l, mid - 1);
build(cur->child[1], cur, mid + 1, r);
cur->updata();
}
} void splay(node *cur, node *goal) {
while(cur->father != goal) {
node *fa = cur->father, *gfa = cur->father->father;
if(gfa != goal) getPos(cur) == getPos(fa) ? rotate(fa) : rotate(cur);
rotate(cur);
}
} node *findKth(ll k) {
node *cur = rt->child[1];
while(true) {
if(k <= cur->child[0]->size) cur = cur->child[0];
else if(k > cur->child[0]->size + cur->right - cur->left + 1) {
k = k - cur->child[0]->size - (cur->right - cur->left + 1);
cur = cur->child[1];
} else {
split(cur, cur->left + k - cur->child[0]->size - 1);
splay(cur, rt);
return cur;
}
}
} ll earse(node *goal) {
node *rplNode = goal->child[0];
while(rplNode->child[1] != nul) rplNode = rplNode->child[1];
if(rplNode != nul) {
splay(rplNode, goal);
cect(rplNode, rt, 1);
cect(goal->child[1], rplNode, 1);
rplNode->updata();
} else cect(goal->child[1], rt, 1);
ll id = goal->left;
delete goal;
return id;
} void insLast(ll k) {
node *cur = rt;
while(cur->child[1] != nul) cur = cur->child[1];
if(cur != rt) splay(cur, rt);
cur->child[1] = newNode(k, k, cur);
cur->size++;
}
} row[MAXN], lastLine; ll leave(ll x, ll y) {
ll id;
if(y != m) {
lastLine.insLast(id = row[x].earse(row[x].findKth(y)));
row[x].insLast(lastLine.earse(lastLine.findKth(x)));
} else lastLine.insLast(id = lastLine.earse(lastLine.findKth(x)));
return id;
} int main() {
init();
ll q, x, y;
scanf("%lld%lld%lld", &n, &m, &q);
lastLine.build(lastLine.rt->child[1], lastLine.rt, 1, n);
for(ll i = 1; i <= n; i++)
(row[i].rt)->child[1] = newNode((i - 1) * m + 1, i * m - 1, row[i].rt);
while(q--) {
scanf("%lld%lld", &x, &y);
printf("%lld\n", leave(x, y));
}
return 0;
}

【NOIP 2017 提高组】列队的更多相关文章

  1. NOIP 2017 提高组 day1t2 时间复杂度

    P3952 时间复杂度 标签 NOIp提高组 2017 时空限制 1000ms / 128MB 小明正在学习一种新的编程语言 A++,刚学会循环语句的他激动地写了好多程序并 给出了他自己算出的时间复杂 ...

  2. NOIP 2008提高组第三题题解by rLq

    啊啊啊啊啊啊今天已经星期三了吗 那么,来一波题解吧 本题地址http://www.luogu.org/problem/show?pid=1006 传纸条 题目描述 小渊和小轩是好朋友也是同班同学,他们 ...

  3. [NOIp 1998 提高组]Probelm 2 连接多位数【2011百度实习生笔试题】

    /*====================================================================== [NOIp 1998 提高组]Probelm 2 连接 ...

  4. NOIP 2014 提高组 题解

    NOIP 2014 提高组 题解 No 1. 生活大爆炸版石头剪刀布 http://www.luogu.org/problem/show?pid=1328 这是道大水题,我都在想怎么会有人错了,没算法 ...

  5. NOIP 2001 提高组 题解

    NOIP 2001 提高组 题解 No 1. 一元三次方程求解 https://vijos.org/p/1116 看见有人认真推导了求解公式,然后猥琐暴力过的同学们在一边偷笑~~~ 数据小 暴力枚举即 ...

  6. 最优贸易 NOIP 2009 提高组 第三题

    题目描述 C 国有 n 个大城市和 m 条道路,每条道路连接这 n 个城市中的某两个城市.任意两个 城市之间最多只有一条道路直接相连.这 m 条道路中有一部分为单向通行的道路,一部分 为双向通行的道路 ...

  7. NOIP 2006 提高组 t1 能量项链

    题目描述 在Mars星球上,每个Mars人都随身佩带着一串能量项链.在项链上有N颗能量珠.能量珠是一颗有头标记与尾标记的珠子,这些标记对应着某个正整数.并且,对于相邻的两颗珠子,前一颗珠子的尾标记一定 ...

  8. [NOIp2017提高组]列队

    [NOIp2017提高组]列队 题目大意 一个\(n\times m(n,m\le3\times10^5)\)的方阵,每个格子里的人都有一个编号.初始时第\(i\)行第\(j\)列的编号为\((i-1 ...

  9. noip 2014 提高组初赛

    noip 2014 提高组初赛 一. TCP协议属于哪一层协议( ) A. 应用层 B. 传输层 C. 网络层 D. 数据链路层 B TCP(传输控制协议) 若有变量int a; float: x, ...

随机推荐

  1. 1923. Scary Politics (timus) (dfs) search

    http://acm.timus.ru/problem.aspx?space=1&num=1923 -- timus This is s problem about thd dfs and s ...

  2. APP专项测试使用到的工具

    最近在读<大话APP测试>,我也就是把需要使用的测试点做一个总结,目前是使用的工具进行的整理,后期慢慢把工具使用案例贴出来

  3. Android(java)学习笔记70:TabActivity使用

    1.首先我们要知道TabActivity是结合TabHost使用的,于是我们自然而然要说明一下TabHost 所谓的TabHost是提供选项卡(Tab页)的窗口视图容器. 此对象包含两个子对象: 一个 ...

  4. [Pytorch] pytorch笔记 <二>

    pytorch笔记2 用到的关于plt的总结 plt.scatter scatter(x, y, s=None, c=None, marker=None, cmap=None, norm=None, ...

  5. 下载Xcode历史版本方法

    1.打开链接:https://developer.apple.com/download/more 进入页面 2.在搜索框输入Xcode,回车搜索.如图,找到各种版本Xcode 搜索Xcode 3.双击 ...

  6. javaweb基础(36)_jdbc进行批处理

    在实际的项目开发中,有时候需要向数据库发送一批SQL语句执行,这时应避免向数据库一条条的发送执行,而应采用JDBC的批处理机制,以提升执行效率. JDBC实现批处理有两种方式:statement和pr ...

  7. webapi中配置返回的时间数据格式

    web api返回的是标准格式UTC时间,如果要转成我们需要的格式,可以在WebApiConfig.cs的Register函数中新增以下配置来定义返回的时间类型格式: //配置返回的时间类型数据格式 ...

  8. jQuery插件的使用和写法

    插件(plugin)也称为扩展(Extension),是一种遵循一定规范的应用程序接口编写出来的程序. jQuery的易扩展性,吸引了来自全球的开发者来共同编写jQuery的插件. jQuery表单验 ...

  9. 谈谈JDK8中的字符串拼接

    字符串拼接问题应该是每个Java程序员都熟知的事情了,几乎每个Java程序员都读过关于StringBuffer/StringBuilder来拼接字符串. 在大多数的教程中,也许你会看到用+号拼接字符串 ...

  10. 泉五培训Day5

    T1 陪审团 题目 [题目描述] 陪审团制度历来是司法研究中的一个热议话题,由于陪审团的成员组成会对案件最终的结果产生巨大的影响,诉讼双方往往围绕陪审团由哪些人组成这一议题激烈争夺.小 W提出了一个甲 ...