题目链接

NOIP2017真的是不按常理出牌:

1、数学题不在Day2T1

2、一道水题一道细节极多的模拟题一道不知道怎么形容的题(小凯的疑惑)(因为我太菜了)

3、3道大火题

当时看到列队这题是毫无头绪的,因为数据大得让你存都存不下,于是果断打了个30分暴力(如果打个离散化还能多骗20分)。

蓦然回首,豁然开朗。

思考可知,一个人出列影响的只是当前一行和最后一列,于是,我们只需要对每一行和最后一列分别用数据结构维护,这个数据结构要支持

1、查询并删除第\(k\)个数。

2、在末尾插入一个数。

若能维护这些,设每次操作的位置为\((x,y)\),那我们只需要把第\(x\)行第\(y\)个数删除并插入到最后一列的末尾去,然后把最后一列第\(x\)个数删除并插入到第\(x\)行的末尾就行了。

接下来考虑用什么数据结构。

\(Splay\)显然是支持这些操作的(事实上几乎所有平衡树都支持,然而我只会常数巨大的\(Splay\))

然后面临的一个问题就是\(n*m\)的范围达到了\((30W)^2\),显然直接开是开不下的。

容易发现\(q\)次操作后,绝大部分人的相对位置是不变的,所以我们可以用点表示区间,查询&删除\(k\)时把这个区间\([l,r]\)分成三部分,

·1、\([l,k-1]\)

·2、\(k\)

·3、\([k+1,r]\)

第一个区间就是原来的点,第三个区间是新加的点,第二个就是我们需要的,直接丢掉,把第一个和第三个点连起来(具体就是把第三部分插入到第一部分和第一部分的右儿子之间)。特别地,如果\(k\)刚好等于这个点维护的区间的两个端点之一,直接把端点维护的区间减一减掉\(k\)就行了。

\(100\)多\(line\)的\(code\),\(debug\)了好久唔唔唔。

#include <cstdio>
#include <cstdlib>
#define ll long long
#define getchar() (S==T&&(T=(S=BB)+fread(BB,1,1<<15,stdin),S==T)?EOF:*S++)
char BB[1<<15],*S=BB,*T=BB;
inline int read(){
int s = 0, w = 1;
char ch = getchar();
while(ch < '0' || ch > '9'){ if(ch == '-') w = -1; ch = getchar();}
while(ch >= '0' && ch <= '9') s = s * 10 + ch - '0', ch = getchar();
return s * w;
}
const int MAXN = 300010;
const int MAX = 10000010;
int n, m, q;
int cnt, size[MAX], son[MAX][2], fa[MAX];
ll L[MAX], R[MAX];
struct Splay_Tree{
int root;
inline void pushup(int x){ size[x] = size[son[x][0]] + size[son[x][1]] + R[x] - L[x] + 1; }
inline int Newnode(ll l, ll r){
L[++cnt] = l; R[cnt] = r;size[cnt] = r - l + 1;
return cnt;
}
inline void init(ll l, ll r){ root = Newnode(l, r); }
inline void rotate(int x){
int y = fa[x]; int z = fa[y]; int k = (son[y][1] == x);
son[z][son[z][1] == y] = x; fa[x] = z;
son[y][k] = son[x][k ^ 1]; fa[son[x][k ^ 1]] = y;
son[x][k ^ 1] = y; fa[y] = x;
pushup(y); pushup(x);
}
inline void Splay(int x, int goal){
if(x == goal) return;
int y, z;
while(fa[x] != goal){
y = fa[x]; z = fa[y];
if(z != goal) (son[z][1] == y) ^ (son[y][1] == x) ? rotate(x) : rotate(y);
rotate(x);
}
if(!goal) root = x;
}
inline void insert(ll x){
int now = root;
while(son[now][1]) now = son[now][1];
int t = Newnode(x, x);
son[now][1] = t; fa[t] = now; Splay(t, 0);
}
inline ll split(int x, ll k){ //分裂点x,删除k
ll s = L[x], t = R[x];
if(k != s && k != t){
int b = Newnode(k + 1, t); R[x] = k - 1;
son[b][1] = son[x][1]; fa[son[x][1]] = b;
son[x][1] = b; fa[b] = x;
Splay(b, 0);
}
else{
if(k == t) --R[x];
else ++L[x];
--size[x];
Splay(x, 0);
}
return k;
}
inline ll popkth(int k){ //删除并返回第k大
int x = root;
while(233){
if(size[son[x][0]] >= k) x = son[x][0];
else {
k -= size[son[x][0]];
if(k <= R[x] - L[x] + 1)
return split(x, k + L[x] - 1);
k -= R[x] - L[x] + 1; x = son[x][1];
}
}
}
inline int build(int l, int r, int f){ //对最后一列的建树
if(l > r) return 0;
int mid = (l + r) >> 1;
int now = Newnode((ll)mid * m, (ll)mid * m);
fa[now] = f;
son[now][0] = build(l, mid - 1, now);
son[now][1] = build(mid + 1, r, now);
pushup(now);
return now;
}
}splay[MAXN];
int a, b;
ll x, y;
int main(){
n = read(); m = read(); q = read();
for(int i = 1; i <= n; ++i) //每行一棵Splay维护前m-1个数
splay[i].init((ll)(i - 1) * m + 1, (ll)i * m - 1); //点表示区间
splay[0].root = splay[0].build(1, n, 0);
for(int i = 1; i <= q; ++i){
a = read(); b = read();
if(b == m){
printf("%lld\n", x = splay[0].popkth(a));
splay[0].insert(x);
continue;
}
printf("%lld\n", x = splay[a].popkth(b));
splay[a].insert(splay[0].popkth(a));
splay[0].insert(x);
}
return 0;
}

[NOIP2017]列队 (Splay)的更多相关文章

  1. Luogu 3960 [NOIP2017] 列队 - splay|线段树

    题解 是我从来没有做过的裂点splay... 看的时候还是很懵逼的QAQ. 把最后一列的$n$个数放在一个平衡树中, 有 $n$ 个点 剩下的$n$行数, 每行都开一个平衡树,开始时每棵树中仅有$1$ ...

  2. 【loj2319】[NOIP2017]列队 Splay(卡过)

    题目描述 给出一个 $n\times m$ 的矩阵,第 $i$ 行第 $j$ 列的数为 $(i-1)\times m+j$ . 现在有 $q$ 次操作,每次操作给出位置 $(x,y)$ ,取出 $(x ...

  3. [NOIP2017]列队 离线+SBT

    [NOIP2017]列队 题目描述 Sylvia 是一个热爱学习的女♂孩子. 前段时间,Sylvia 参加了学校的军训.众所周知,军训的时候需要站方阵. Sylvia 所在的方阵中有n×m名学生,方阵 ...

  4. 题解[NOIP2017] 列队

    题解[NOIP2017] 列队 题面 解析 看到这题时感觉这个编号很难维护啊? 后来看了lzf大佬的题解才会.. 首先,考虑一个稍微暴力的做法, 维护每一行的前\(m-1\)个人和最后一列的\(n\) ...

  5. 【NOIP2017】列队 splay

    当年太菜了啊,连$60$分的暴力都没拿满,只打了一个$30$分的. 考虑到这题最多只会询问到$30W$个点,且整个矩阵会去到$30W\times 30W$,显然不能将所有的点存下来. 对于每一行(除最 ...

  6. [NOIP2017]列队(线段树/裂点splay)

    考虑n=1的做法,就是支持: 1.在线删一个数 2.在结尾加一个数 3.查询序列的第y个数 用线段树记录区间内被删元素的个数,可以通过线段树上二分快速得解,对于新增的数,用vector记录即可. 对于 ...

  7. Luogu3960 NOIP2017列队(splay/线段树)

    令splay中的一个点表示一段区间,需要使用其中某个点时将区间分裂即可,剩下的都是splay的基本操作了.写的非常丑陋,注意细节.感觉考场上肯定只能靠部分分苟活了.想起来去年因为各种莫名其妙的原因50 ...

  8. NOIP2017列队(phalanx)解题报告

    列队作为NOIP2017最后一道题,其实并不难,只是相对于其它题目,有点小小的工业 首先,这道题我用splay维护的,如果你不会splay,又想学一下splay,可以来这里学一学,接下来步入正题 首先 ...

  9. NOIP2017 列队 题解报告【56行线段树】

    题目描述 Sylvia 是一个热爱学习的女♂孩子. 前段时间,Sylvia 参加了学校的军训.众所周知,军训的时候需要站方阵. Sylvia 所在的方阵中有n \times mn×m名学生,方阵的行数 ...

随机推荐

  1. Eclipse中JS文件红叉处理

    使用新版本的Eclipse 或者 MyEclipse,项目中的 JS文件出现红叉,让人觉得项目中存在错误代码,给人的感觉很不爽. 记录一下去掉红叉的方法: 第1步: 打开工作空间中的项目找到项目的 . ...

  2. Mootools 学习随笔

    简单的介绍下Mootools: MooTools是一个简洁,模块化,面向对象的开源JavaScript web应用框架.在处理js.css.html时候,为web开发者提供了一个跨浏览器的js解决方案 ...

  3. Django admin源码剖析

    单例模式 单例模式(Singleton Pattern)是一种常用的软件设计模式,该模式的主要目的是确保某一个类只有一个实例存在.当你希望在整个系统中,某个类只能出现一个实例时,单例对象就能派上用场. ...

  4. 把python脚本打包成win可执行文件

    前几天有个朋友找我写一点小东西,写好后把代码发他帮他搞了半天,结果愣是没听懂,就找到了这个办法. 1.导入pyinstaller包, pip install pyinstaller 2.进入到你需要打 ...

  5. git 创建分支并提交到服务器对应的新分支

    1.切换到源分支 git checkout test 2.在源分支的基础上创建新分支 git branch test1 3.提交到远程分支 git pull 会自动提示下面的命令 git pull - ...

  6. restFul介绍及其使用规范

    什么是REST和RESTful API? REST:(英文:Representational State Transfer,简称REST)表征性状态转移,是一种软件架构风格. RESTful : RE ...

  7. ajax中用jsonp接收json数据

    最近在做查快递网页时遇到一个问题,调用的快递100的api,但是快递100api不允许跨域请求,就是用127.0.0.1发的请求会直接被拦截. 只是个简单的网页,不想自己做服务器转发,最后找到了一个y ...

  8. ORACLE和SQL语法区别归纳

    数据类型比较类型名称 Oracle   SQLServer   比较字符数据类型  CHAR  CHAR  都是固定长度字符资料但oracle里面最大度为2kb,SQLServer里面最大长度为8kb ...

  9. hadoop 2.6.0 伪分布式部署安装遇到的问题

    之前读到了一篇关于配置安装hadoop的博文(地址:http://www.powerxing.com/install-hadoop/)能正确安装和运行,但是在网页进行Jobtracker监控时,输入l ...

  10. Intellij Idea 创建Web项目入门

    相关软件: Intellij Idea14:http://pan.baidu.com/s/1nu16VyD JDK7:http://pan.baidu.com/s/1dEstJ5f Tomcat(ap ...