题目大意:有一个$n \times m$的方阵,第$i$行第$j$列的人的编号是$(i-1) \times m + j$。

现在有$q$个出列操作,每次让一个人出列,然后让这个人所在行向左看齐,再让最后一列向前看齐,最后让这个人站到第$n$行第$m$列的位置。

你需要输出每次出列的人的编号。

题解:可以每行维护一棵平衡树,再给最后一列维护一棵平衡树(虽然正解是用树状数组)。

发现每行的人初始编号是连续的,而对于$9 \times 10 ^ {10}$的人数,$3 \times 10 ^ {5}$次询问改变的人数其实并不多。

所以,我们把同一行内编号连续的一段缩成一个点,然后需要出列时再分裂即可。

卡点:因为正解是树状数组,所以我洛谷被玄学卡$RE$($UOJ$过的),数组大小调了好几次,然后发现开$O(2)$,数组开的比$UOJ$大一些就不$RE$了。。。

C++ Code:(洛谷)

// luogu-judger-enable-o2
#include <cstdio>
#include <cstdlib>
#define maxn 5500010
using namespace std;
struct node{
int l, r;
long long head;
void operator -= (long long a) {l -= a; r -= a;}
} val[maxn];
int n, m, q;
int lc[maxn], rc[maxn], num[maxn], sz[maxn], tg[maxn], idx;
struct treap {
int root, len;
int ta, tb, tmp, tmp6;
int nw(node x) {
val[++idx] = x;
sz[idx] = 1;
num[idx] = rand();
lc[idx] = rc[idx] = 0;
return idx;
}
void update(int p) {sz[p] = sz[lc[p]] + sz[rc[p]] + 1;}
void pushdown(int p) {
tmp6 = tg[p]; tg[p] = 0;
val[lc[p]] -= tmp6;
val[rc[p]] -= tmp6;
tg[lc[p]] += tmp6;
tg[rc[p]] += tmp6;
}
void splitl(int rt, int k, int &x, int &y) {
if (!rt) x = y = 0;
else {
if (val[rt].l <= k) splitl(rc[rt], k, rc[rt], y), x = rt;
else splitl(lc[rt], k, x, lc[rt]), y = rt;
update(rt);
}
}
void splitr(int rt, int k, int &x, int &y) {
if (!rt) x = y = 0;
else {
pushdown(rt);
if (val[rt].r <= k) splitr(rc[rt], k, rc[rt], y), x = rt;
else splitr(lc[rt], k, x, lc[rt]), y = rt;
update(rt);
}
}
int merge(int x, int y) {
if (!x || !y) return x | y;
if (num[x] > num[y]) {
pushdown(x);
rc[x] = merge(rc[x], y);
update(x);
return x;
} else {
pushdown(y);
lc[y] = merge(x, lc[y]);
update(y);
return (y);
}
}
void build(node p) {
root = nw(p);
}
void insert(long long p, int k = -1){
if (k == -1) k = len;
if (!root) root = nw((node) {k, k, p});
else root = merge(root, nw((node) {k, k, p}));
}
long long nxt(int k) {
splitr(root, k - 1, ta, tmp);
splitl(tmp, k, tmp, tb);
int a = nw((node) {val[tmp].l, k - 1, val[tmp].head}),
b = nw((node) {k + 1, val[tmp].r, val[tmp].head + k + 1 - val[tmp].l});
tb = merge(b, tb);
tg[tb]++;
val[tb] -= 1;
root = merge(merge(ta, a), tb);
return k - val[tmp].l + val[tmp].head;
}
} s[300010], back;
int main() {
srand(20040826);
scanf("%d%d%d", &n, &m, &q);
back.len = n;
for (long long i = 1; i <= n; i++) back.insert(i * m, i);
for (long long i = 1; i <= n; i++) {
s[i].len = m - 1;
s[i].build((node){1, m - 1, (i - 1) * m + 1});
}
int x, y;
long long ta, tb;
while (q--) {
scanf("%d%d", &x, &y);
if (y != m) {
ta = s[x].nxt(y);
tb = back.nxt(x);
s[x].insert(tb);
back.insert(ta);
} else {
ta = back.nxt(x);
back.insert(ta);
}
printf("%lld\n", ta);
}
return 0;
}

  

C++ Code:(UOJ)

#include <cstdio>
#include <cstdlib>
#define maxn 5000010
using namespace std;
struct node{
int l, r;
long long head;
void operator -= (long long a) {l -= a; r -= a;}
} val[maxn];
int n, m, q;
int lc[maxn], rc[maxn], num[maxn], sz[maxn], tg[maxn], idx;
struct treap {
int root, len;
int ta, tb, tmp;
int nw(node x) {
val[++idx] = x;
sz[idx] = 1;
num[idx] = rand();
lc[idx] = rc[idx] = 0;
return idx;
}
void update(int p) {sz[p] = sz[lc[p]] + sz[rc[p]] + 1;}
void pushdown(int p) {
int tmp6 = tg[p]; tg[p] = 0;
val[lc[p]] -= tmp6;
val[rc[p]] -= tmp6;
tg[lc[p]] += tmp6;
tg[rc[p]] += tmp6;
}
void splitl(int rt, int k, int &x, int &y) {
if (!rt) x = y = 0;
else {
if (val[rt].l <= k) splitl(rc[rt], k, rc[rt], y), x = rt;
else splitl(lc[rt], k, x, lc[rt]), y = rt;
update(rt);
}
}
void splitr(int rt, int k, int &x, int &y) {
if (!rt) x = y = 0;
else {
pushdown(rt);
if (val[rt].r <= k) splitr(rc[rt], k, rc[rt], y), x = rt;
else splitr(lc[rt], k, x, lc[rt]), y = rt;
update(rt);
}
}
int merge(int x, int y) {
if (!x || !y) return x | y;
if (num[x] > num[y]) {
pushdown(x);
rc[x] = merge(rc[x], y);
update(x);
return x;
} else {
pushdown(y);
lc[y] = merge(x, lc[y]);
update(y);
return (y);
}
}
void build(node p) {
root = nw(p);
}
void insert(long long p, int k = -1){
if (k == -1) k = len;
if (!root) root = nw((node) {k, k, p});
else root = merge(root, nw((node) {k, k, p}));
}
long long nxt(int k) {
splitr(root, k - 1, ta, tmp);
splitl(tmp, k, tmp, tb);
int a = nw((node) {val[tmp].l, k - 1, val[tmp].head}),
b = nw((node) {k + 1, val[tmp].r, val[tmp].head + k + 1 - val[tmp].l});
tb = merge(b, tb);
tg[tb]++;
val[tb] -= 1;
root = merge(merge(ta, a), tb);
return k - val[tmp].l + val[tmp].head;
}
} s[300010], back;
int main() {
srand(20040826);
scanf("%d%d%d", &n, &m, &q);
back.len = n;
for (long long i = 1; i <= n; i++) back.insert(i * m, i);
for (long long i = 1; i <= n; i++) {
s[i].len = m - 1;
s[i].build((node){1, m - 1, (i - 1) * m + 1});
}
int x, y;
long long ta, tb;
while (q--) {
scanf("%d%d", &x, &y);
if (y != m) {
ta = s[x].nxt(y);
tb = back.nxt(x);
s[x].insert(tb);
back.insert(ta);
} else {
ta = back.nxt(x);
back.insert(ta);
}
printf("%lld\n", ta);
}
return 0;
}

  

[NOIP2017 TG D2T3]列队的更多相关文章

  1. 【NOIP题解】NOIP2017 TG D2T3 列队

    列队,NOIP2017 TG D2T3. 树状数组经典题. 题目链接:洛谷. 题意: Sylvia 是一个热爱学习的女孩子. 前段时间,Sylvia 参加了学校的军训.众所周知,军训的时候需要站方阵. ...

  2. [NOIp2017提高组]列队

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

  3. 【学术篇】NOIP2017 d2t3 列队phalanx splay做法

    我可去他的吧.... ==============先胡扯些什么的分割线================== 一道NOIP题我调了一晚上...(其实是因为昨晚没有找到调试的好方法来的说...) 曾经我以 ...

  4. NOIP2017 D2T3列队

    这题我改了三天,考场上部分分暴力拿了50,考完试发现与正解很接近只是没写出来. 对于每一行和最后一列建n+1颗线段树,维护前缀和. 复杂度qlogn 假如你移动一个坐标为(x,y)的人,你要将第x行线 ...

  5. noip2017 TG 游记

    嗨小朋友们大家好,还记得我是谁吗?我就是为GG代言的蒟蒻--xzz 今天呐我特别的要向HN的dalao们ZJ的巨佬们还有全国的神犇们问声好 为什么呢因为我们在2017年11月份来到了吔屎的长沙理工大学 ...

  6. NOIP2017 Day2 T3 列队(treap)

    可以直接用treap上大模拟...n+1个treap维护n行的前m-1个点和最后一列. 需要支持删除一个点或者一段区间,而空间并不支持存下所有的点的时候,可以用一个点代替一个区间,记录区间首项的值和区 ...

  7. [NOIP2015 TG D2T3]运输计划

    题目大意: 给你一棵n个节点的树,有边权,有多个任务,每个要求从ui号节点到 vi号节点去.m 个计划, 这 m 个计划会同时开始.当这 m 个任务都完成时,工作完成. 现在可以把任意一个边的边权变为 ...

  8. [NOIP2016 TG D2T3]愤怒的小鸟

    题目大意:有一架弹弓位于(0,0)处,每次可以用它向第一象限发射一只小鸟,飞行轨迹均为形如y=ax2+bxy=ax+bx2 y=ax2+bx的曲线,且必须满足a<0(即是下开口的) 平面的第一象 ...

  9. [NOIP2017 TG D2T2]宝藏(模拟退火)

    题目大意:$NOIPD2T2$宝藏 题解:正常做法:状压DP .这次模拟退火,随机一个排列,$O(n^2)$贪心按排列的顺序加入生成树 卡点:没开$long\;long$,接受较劣解时判断打错,没判$ ...

随机推荐

  1. BootStrap的动态模态框及静态模态框

    1.要用bootStrap这个框架就必须要重载它的class类,也就是说class要一样 代码如下: 有疑问的可以在下面留言,欢迎大家一起交流 1.1动态模态框 <!DOCTYPE html&g ...

  2. IDEA项目启动报Unable to open debugger port (127.0.0.1:51554): java.net.SocketException "socket closed"

    启动报错: Unable to open debugger port (127.0.0.1:51554): java.net.SocketException "socket closed&q ...

  3. 20.2 解析与序列化【JavaScript高级程序设计第三版】

    JSON 之所以流行,拥有与JavaScript 类似的语法并不是全部原因.更重要的一个原因是,可以把JSON 数据结构解析为有用的JavaScript 对象.与XML 数据结构要解析成DOM 文档而 ...

  4. 02---Nginx

    Nginx .caret, .dropup > .btn > .caret { border-top-color: #000 !important; } .label { border: ...

  5. Javaweb——四则运算---18.11.01

    ---恢复内容开始--- test.jsp <%@ page language="java" contentType="text/html; charset=utf ...

  6. WPF中使用第三方字体选择器

    原文:WPF中使用第三方字体选择器 起因 到WPF的字体可以设置的东西变得非常的多,而却没有提供专用的字体选择对话框,甚至于WinFrom的FontDialog也是不能直接用来设置WPF中的字体.解决 ...

  7. 创龙DSP6748的DAC例程研究

    1. 创龙DSP6748开发板驱动TL5724这个DAC,输出指定的电压值,此程序是使用 IO 口模拟 SPI 实现与 TL5724 模块的数据交互. 2. 首先是初始化PSC函数 void PSCI ...

  8. Word 2013 无法撤销操作的错误

    来自 <http://delxu.blog.51cto.com/975660/1409139> 关闭正在运行的所有程序. 按Win-R,在运行框中键入regedit,然后单击“确定”. 在 ...

  9. redux使用过程中遇到的两个致命的关键点

    一.在reducer中,返回的state必须是全新的对象,否则,redux不会执行listening方法,因为redux会认为state没有更新过,没必要重新渲染view. 出现问题的例子: cons ...

  10. 拉普拉斯矩阵(Laplacian Matrix) 及半正定性证明

    摘自 https://blog.csdn.net/beiyangdashu/article/details/49300479 和 https://en.wikipedia.org/wiki/Lapla ...