[NOIP2017 TG D2T3]列队
题目大意:有一个$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]列队的更多相关文章
- 【NOIP题解】NOIP2017 TG D2T3 列队
列队,NOIP2017 TG D2T3. 树状数组经典题. 题目链接:洛谷. 题意: Sylvia 是一个热爱学习的女孩子. 前段时间,Sylvia 参加了学校的军训.众所周知,军训的时候需要站方阵. ...
- [NOIp2017提高组]列队
[NOIp2017提高组]列队 题目大意 一个\(n\times m(n,m\le3\times10^5)\)的方阵,每个格子里的人都有一个编号.初始时第\(i\)行第\(j\)列的编号为\((i-1 ...
- 【学术篇】NOIP2017 d2t3 列队phalanx splay做法
我可去他的吧.... ==============先胡扯些什么的分割线================== 一道NOIP题我调了一晚上...(其实是因为昨晚没有找到调试的好方法来的说...) 曾经我以 ...
- NOIP2017 D2T3列队
这题我改了三天,考场上部分分暴力拿了50,考完试发现与正解很接近只是没写出来. 对于每一行和最后一列建n+1颗线段树,维护前缀和. 复杂度qlogn 假如你移动一个坐标为(x,y)的人,你要将第x行线 ...
- noip2017 TG 游记
嗨小朋友们大家好,还记得我是谁吗?我就是为GG代言的蒟蒻--xzz 今天呐我特别的要向HN的dalao们ZJ的巨佬们还有全国的神犇们问声好 为什么呢因为我们在2017年11月份来到了吔屎的长沙理工大学 ...
- NOIP2017 Day2 T3 列队(treap)
可以直接用treap上大模拟...n+1个treap维护n行的前m-1个点和最后一列. 需要支持删除一个点或者一段区间,而空间并不支持存下所有的点的时候,可以用一个点代替一个区间,记录区间首项的值和区 ...
- [NOIP2015 TG D2T3]运输计划
题目大意: 给你一棵n个节点的树,有边权,有多个任务,每个要求从ui号节点到 vi号节点去.m 个计划, 这 m 个计划会同时开始.当这 m 个任务都完成时,工作完成. 现在可以把任意一个边的边权变为 ...
- [NOIP2016 TG D2T3]愤怒的小鸟
题目大意:有一架弹弓位于(0,0)处,每次可以用它向第一象限发射一只小鸟,飞行轨迹均为形如y=ax2+bxy=ax+bx2 y=ax2+bx的曲线,且必须满足a<0(即是下开口的) 平面的第一象 ...
- [NOIP2017 TG D2T2]宝藏(模拟退火)
题目大意:$NOIPD2T2$宝藏 题解:正常做法:状压DP .这次模拟退火,随机一个排列,$O(n^2)$贪心按排列的顺序加入生成树 卡点:没开$long\;long$,接受较劣解时判断打错,没判$ ...
随机推荐
- Deprecated: mysql_connect(): The mysql extension is deprecated and will be removed in the future: use mysqli or PDO
你有碰上过这样的提示吗? Deprecated: mysql_connect(): The mysql extension is deprecated and will be removed in t ...
- connect() to unix:/var/run/php-fpm.sock failed (11: Resource temporarily unavailable)
nginx + php做服务,在高并发的时候会出现一些错误 connect() to unix:/var/run/php-fpm.sock failed (11: Resource temporar ...
- 一笔画问题 南阳acm42(貌似没用到什么算法)
一笔画问题 时间限制:3000 ms | 内存限制:65535 KB 难度:4 描述 zyc从小就比较喜欢玩一些小游戏,其中就包括画一笔画,他想请你帮他写一个程序,判断一个图是否能够用一笔画下 ...
- 2-Linux C语言指针与内存-学习笔记
Linux C语言指针与内存 前面我们对于: c语言的基本用法 makeFile文件的使用 main函数的详解 标准输入输出流以及错误流管道 工具与原理 指针与内存都是c语言中的要点与难点 指针 数组 ...
- Python3爬虫(一)HTTP相关基础
Infi-chu: http://www.cnblogs.com/Infi-chu/ 一.URI.URL.URN.HTTP URI:统一资源标志符 URL:是URI的一个子集 URN:是URI的另一个 ...
- python2.7入门---字典(Dictionary)
这次咱们记录的是python中的字典这个鬼,首先我们得了解,字典是另一种可变容器模型,且可存储任意类型对象.字典的每个键值 key=>value 对用冒号 : 分割,每个键值对之间用逗号 ...
- Android面试收集录 数据库
1.SQLite数据库如何查询表table1的第20条到30条记录? select * from table1 limit 19,11 ==>从19开始,11个数据 2.如何才能将table ...
- 联想ThinkPad S3-S440虚拟机安装,ubuntu安装,Hadoop(2.7.1)详解及WordCount运行,spark集群搭建
下载ubuntu操作系统版本 ubuntu-14.10-desktop-amd64.iso(64位) 安装过程出现错误: This kernel requires an X86-64 CPU,but ...
- Linux篇:因为修改了/etc/sudoers 文件的权限导致的问题
因为想要把sudo变成免密码所以就查了网上的教程.说是要修改/etc/sudoers文件,但是修改的时候发现这个文件是只读, 所以就 /etc/sudoers 结果就导致了接下来用sudo的时候提示如 ...
- Hibernate-ORM:08.Hibernate中的投影查询
------------吾亦无他,唯手熟尔,谦卑若愚,好学若饥------------- 本篇博客将叙述hibernate中的投影查询 一,目录: 1.解释什么是投影查询 2.返回Object单个对象 ...