题意简述

有一个n × m 的矩阵,第i行第j列元素编号为(i - 1)× m +j

每次将一个数取出,其他元素依次向左,向上填补空缺,最后将取出的数放入矩阵最后一格

求每次取出数的编号

题解思路

由于最后一列较为特殊,只有当取出的数位于最后一列,向上填补空缺时才有影响,所有特殊考虑

对于每一行的(m - 1)个元素 和 最后一列元素都维护一个splay

进行删除第k个元素 和 在末尾插入元素 两个操作

代码

#include <cstdio>
#include <cassert>
using namespace std;
typedef long long ll;
struct Node
{
ll l, r;
Node *c[2];
int s;
Node() {l = r = 0; s = 0; c[0] = c[1] = 0; }
Node(ll x) {l = r = x; s = 1; c[0] = c[1] = 0; }
Node(ll l, ll r) {this -> l = l; this -> r = r; s = r - l + 1; c[0] = c[1] = 0; }
int get(bool b){return c[b] ? c[b] -> s : 0; }
void upt(){s = get(0) + get(1) + (r - l + 1); }
int cmp(int x){return (x > get(0) && x <= get(0) + r - l + 1) ? -1 : x > get(0); }
}*r[300010];
int n, m, q, x, y;
void rtt(Node* &o, bool b)
{
Node* t = o -> c[b ^ 1];
o -> c[b ^ 1] = t -> c[b];
t -> c[b] = o;
o -> upt();
t -> upt();
o = t;
}
void splay(Node* &o, int k)
{
if (!o) return;
int d1 = o -> cmp(k);
if (d1 == -1) return;
int xx = k - d1 * (o -> s - o -> get(1));
int d2 = o -> c[d1] -> cmp(xx);
if (d2 == -1) rtt(o, d1 ^ 1);
else
{
splay(o -> c[d1] -> c[d2], xx - d2 * (o -> c[d1] -> s - o -> c[d1] -> get(1)));
if (d1 == d2) rtt(o, d2 ^ 1);
else rtt(o -> c[d1], d2 ^ 1);
rtt(o, d1 ^ 1);
}
}
void ins(Node* &o, Node* x)
{
if (!o) {o = x; return; }
splay(o, o -> s);
o -> c[1] = x;
o -> upt();
}
Node* del(Node* &o, int k)
{
splay(o, k);
if (o -> l == o -> r)
{
Node* t = o;
if (!o -> c[0]) o = o -> c[1];
else
{
splay(o -> c[0], o -> get(0));
o -> c[0] -> c[1] = o -> c[1];
o = o -> c[0];
o -> upt();
}
t -> s = 1;
t -> c[0] = t -> c[1] = 0;
return t;
}
ll xxx = o -> l + k - o -> get(0) - 1;
Node* x = new Node(o -> l, xxx - 1);
o -> l = xxx + 1;
x -> c[0] = o -> c[0];
o -> c[0] = x;
x -> upt();
o -> upt();
return new Node(xxx);
}
int main()
{
scanf("%d%d%d", &n, &m, &q);
for (register int i = 1; i <= n; ++i)
{
r[i] = new Node((ll)(i - 1) * m + 1, (ll)i * m - 1);
ins(r[0], new Node((ll)i * m));
}
for (register int i = 1; i <= q; ++i)
{
scanf("%d%d", &x, &y);
Node* idx;
if (y == m)
{
printf("%lld\n", (idx = del(r[0], x)) -> l);
ins(r[0], idx);
}
else
{
ins(r[x], del(r[0], x));
printf("%lld\n", (idx = del(r[x], y)) -> l);
ins(r[0], idx);
}
}
}

洛谷 P1960 列队的更多相关文章

  1. 洛谷P3960 列队(NOIP2017)(Splay)

    洛谷题目传送门 最弱的Splay...... 暴力模拟30分(NOIP2017实际得分,因为那时连Splay都不会)...... 发现只是一个点从序列里搬到了另一个位置,其它点的相对位置都没变,可以想 ...

  2. 洛谷 P3960 列队 解题报告

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

  3. 洛谷P3960 列队 NOIp2017 线段树/树状数组/splay

    正解:动态开点线段树 解题报告: 传送门! 因为最近学主席树的时候顺便get到了动态开点线段树?刚好想起来很久很久以前就想做结果一直麻油做的这题,,,所以就做下好了QAQ 然后说下,这题有很多种方法, ...

  4. 洛谷P3960 列队(动态开节点线段树)

    题意 题目链接 Sol 看不懂splay..,看不懂树状数组... 只会暴力动态开节点线段树 观察之后不难发现,我们对于行和列需要支持的操作都是相同的:找到第\(k\)大的元素并删除,在末尾插入一个元 ...

  5. 洛谷P3960 列队(Splay)

    传送门 感觉自己好久不打数据结构已经完全不会了orz…… 据说正解树状数组?然而并不会 首先考虑一下每一次操作,就是把一个人从这一行中取出并放到行的最后,再从最后一列取出放到列的最后 那么这两种操作其 ...

  6. 洛谷 P3960 列队【线段树】

    用动态开点线段树分别维护每一行和最后一列,线段树的作用是记录被选的点的个数以及查询第k个没被选的点,每次修改,从行里标记被选的点,从最后一列标记向左看齐之后少的点,然后用vector维护行列的新增点 ...

  7. 洛谷 P3960 列队

    https://www.luogu.org/problemnew/show/P3960 常数超大的treap #pragma GCC optimize("Ofast") #incl ...

  8. 洛谷P3960 列队 Splay

    其实思路并不算太难,就是代码量相对较大. 我们将一次离队转换为一次删除和两次加入,这样就可以保证空间是动态分配的,最大也不会暴空间. 说实话写之前感觉会很恶心,但是代码量就还好吧,有些细节需要特殊注意 ...

  9. 【洛谷P3960】列队题解

    [洛谷P3960]列队题解 题目链接 题意: Sylvia 是一个热爱学习的女孩子. 前段时间,Sylvia 参加了学校的军训.众所周知,军训的时候需要站方阵. Sylvia 所在的方阵中有 n×m ...

随机推荐

  1. .net core2学习笔记

    在Linux上安装完netcore的sdk后,发现每次重新登录dotnet命令都会失效,咨询完同事后才知道之前的设置只是临时变量,需要vim /etc/profile   编辑这个文件,把环境变量写入 ...

  2. CDQZ 集训大总结

    好爆炸的一次集训…… 成绩: 什么鬼, 烂到一定地步了. 在这里每天考试80%都是暴力,正解思维难度的确比之前大了很多,考的范围也扩大了,比起之前的单独考一个知识点,转变为了多知识点多思维的综合,见了 ...

  3. Linux关闭进程。

    一.shell命令根据端口后关闭指定进程. $(netstat -nlp | | awk '{print $7}' | awk -F"/" '{ print $1 }') nets ...

  4. 1.jdk1.7到jdk1.8 Map发生了什么变化(底层)?

    1.8之后hashMap的数据结构发生了变化,从之前的单纯的数组+链表结构变成数组+链表+红黑树.也就是说在JVM存储hashMap的K-V时仅仅通过key来决定每一个entry的存储槽位(Node[ ...

  5. stack用法,queue用法,

    stack stack 模板类的定义在头文件中. stack 模板类需要两个模板参数,一个是元素类型,一个容器类型,但只有元素类型是必要 的,在不指定容器类型时,默认的容器类型为deque. 定义st ...

  6. [二次编码,数据类型补充以及各种坑]https://i.cnblogs.com/EditPosts.aspx?postid=11184330

    数据类型补充 str:不可变数据类型 1.capitalize首字母大写 name="song" n=name.capitalize() print(n) Song 2.title ...

  7. python 中_init_函数以及参数self

    1)class类包含: 类的属性:类中所涉及的变量 类的方法:类中函数 2)_init_函数(方法) 1.首先说一下,带有两个下划线开头的函数是声明该属性为私有,不能在类地外部被使用或直接访问. 2. ...

  8. RecyclerView下拉加载集合越界问题

    问题描述 在做毕业设计app中遇到这样一个问题,使用RecyclerView进行下拉加载数据的时候,比如我每次让它加载5条数据,当服务器端数据总数刚好是5的倍数的时候,不会出现下拉加载数据集合越界的问 ...

  9. C#5.0新增功能01 异步编程

    连载目录    [已更新最新开发文章,点击查看详细] 如果需要 I/O 绑定(例如从网络请求数据或访问数据库),则需要利用异步编程. 还可以使用 CPU 绑定代码(例如执行成本高昂的计算),对编写异步 ...

  10. 【微信小程序】微信小程序-实现tab

    一.前言 小程序开发中,有很多封装好的控件供开发者使用,但是,很常见的tab选项卡居然没有,只能自己搞一个. 实现原理也很简单,无非是用给view(tab)设置一个点击事件bintap,并且给view ...