[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$,接受较劣解时判断打错,没判$ ...
随机推荐
- 关于okHttp框架的使用
在之前的项目中,使用传统的HttpClient来返回一个图片信息流的时候总是报错,最后发现是因为传统的传输方式会对流的大小有限制,当超过某个值的时候就会报异常,最后决定使用OkHttp框架来解决这个问 ...
- 三、css篇
#这里强烈推荐一本书<css世界>,css第一书. #上面的层叠顺序得记住. 1.align-items justify-content 是flex(弹性盒模型)必须要会的属性,alig ...
- dedesmc 手机端生成静态页
dedesmc 手机端生成静态页 1.首先下载插件,下载地址:https://pan.baidu.com/s/1Nfx_KBYuxRkZ7VzoPxy28g 密码:83x7 2.进入 dedecms ...
- 电子商城实录------定义init初始化的方法
路由方法的设置 //路由方法 private static function dispatch(){ //获取控制器名称(类比:英文单词的后缀) $controller_name=CONTROLLER ...
- php开发aes加密总结
<?php class Aes { /** * aes 加密 解密类库 * @by singwa * Class Aes *说明:本类只适用于加密字符串 * */ private $key = ...
- TP3.2.3 接入阿里sms 短信接口
阿里云短信接口 配置文件 config.php //阿里大鱼 'Ali_SMS' =>array( 'sms_temp' =>'短信模板', 'sms_sign' =>'签名', ' ...
- ERROR: bootstrap checks failed
错误描述:Linux默认配置的参数过小,需要自己设置 max file descriptors [4096] for elasticsearch process is too low, increas ...
- 【Leetcode】Jewels and Stones
Jewels and Stones Description You're given strings J representing the types of stones that are jewel ...
- java_hdfs之读写文件
package hdfsTest.answer.hdfs; import java.io.IOException; import java.net.URI; //import java.net.URL ...
- mvc4 Forms验证存储 两种登录代码
自己也不知道网上看到的第一种居多,第二种用到的人很少,第二种代码十分简洁,就是不清楚是否有安全隐患. 要采用Forms身份验证,先要在应用程序根目录中的Web.config中做相应的设置: <a ...