洛谷P1160 队列安排
洛谷 P1160 队列安排
题目描述
一个学校里老师要将班上NN个同学排成一列,同学被编号为1\sim N1∼N,他采取如下的方法:
- 先将11号同学安排进队列,这时队列中只有他一个人;
- 2-N2−N号同学依次入列,编号为i的同学入列方式为:老师指定编号为i的同学站在编号为1\sim (i -1)1∼(i−1)中某位同学(即之前已经入列的同学)的左边或右边;
- 从队列中去掉M(M<N)M(M<N)个同学,其他同学位置顺序不变。
在所有同学按照上述方法队列排列完毕后,老师想知道从左到右所有同学的编号。
输入格式
第11行为一个正整数NN,表示了有NN个同学。
第2-N2−N行,第ii行包含两个整数k,pk,p,其中kk为小于ii的正整数,pp为00或者11。若pp为00,则表示将ii号同学插入到kk号同学的左边,pp为11则表示插入到右边。
第N+1N+1行为一个正整数MM,表示去掉的同学数目。
接下来MM行,每行一个正整数xx,表示将xx号同学从队列中移去,如果xx号同学已经不在队列中则忽略这一条指令。
输出格式
11行,包含最多NN个空格隔开的正整数,表示了队列从左到右所有同学的编号,行末换行且无空格。
输入输出样例
输入 #1复制
1 | 4 |
输出 #1复制
1 | 2 4 1 |
说明/提示
样例解释:
将同学22插入至同学11左边,此时队列为:
2 121
将同学33插入至同学22右边,此时队列为:
2 3 1231
将同学44插入至同学11左边,此时队列为:
2 3 4 12341
将同学33从队列中移出,此时队列为:
2 4 1241
同学33已经不在队列中,忽略最后一条指令
最终队列:
2 4 1241
数据范围
对于20%20%的数据,有N≤10N≤10;
对于40%40%的数据,有N≤1000N≤1000;
对于100%100%的数据,有N, M≤100000N,M≤100000。
分析
参考https://www.luogu.com.cn/blog/onlynagesha/solution-p1160
用了vector试了一下,果不其然TLE。通过题解学到了一个新的STL list。list是一个双向链表,可以在头尾插入元素。
1 | myList.push_front(1); |
对应的有pop_front()用于移除头部的元素,pop_back()用于移除尾部的元素。
1 | typedef list<int>::iterator Iter; |
这段代码演示的是list提供的,用于访问内部元素的迭代器。迭代器的类型是list
迭代器的用法和指针有些像,可以用*运算符访问内部的元素,++和–运算符可以将它后移或前移一位(建议写成前置形式),用==和!=运算符进判断两个迭代器所指的位置是否一致。但要注意:list的迭代器不支持it += x或it1 - it2这样的运算,也不支持<,<=等运算符。
回到这道题上,最主要的就是要记录每个元素的位置,因为是链表,所以在插入或是删除元素后,其他的元素不会变动,所以可以直接用一个迭代器数组list<int> pos[MAXN]
来保存每个元素的位置,方便之后的插入和删除。
代码
1 |
|
洛谷P1160 队列安排