【数据结构】维护队列

T69293 维护队列

题目描述

Alice 给 Bob 布置了很多工作,他忙的不可开交,决定按照“先进先出(FIFO)”的顺序依次处理这些工作。但是处理过程中,Bob 意识到这种顺序可能不是最优的,因此他会选择性的把某些工作延后。

抽象来说,你需要维护一个队列,支持三种操作:

操作一:1 v,在队尾加入一个价值为 v 的任务。

操作二:2,如果当前队列为空,输出 -1;否则输出队头任务的价值,并将队头弹出。

操作三:3,如果当前队列为空,则不进行任何操作;否则将队列中价值最小的任务挪到队尾,保证没有价值相等的任务。

输入输出格式

输入格式:

第一行一个整数 n,表示操作个数。

接下来 n 行,每行一个或两个整数,格式如上。

输出格式:

对于每个操作 2,输出答案。

输入输出样例

输入样例#1:

5
1 1
1 2
2
2
2

输出样例#1:

1
2
-1

输入样例#2:

6
1 1
1 2
3
2
2
2

输出样例#2:

2
1
-1

【题解】:听过讲解之后,发现这个想法非常有趣。其实Push,Pop一般队列都可以维护,但是对于Mov操作就需要大家开动一下脑筋了。其实Mov不一定需要移动。需要的是标记,最小值标记,然后用Set来查找最小值的位置,然后Push进去的历史编号。得到的Val[No] 用数组来保存历史编号的值即可。Push( ) 进入队列不仅仅是队列,还有Set。Pop( ) 弹出时注意,有些用Mov标记的可能出现在队头,记得要把它弹出。Mov( )操作涉及到两个问题。第一个:用Set的头元素即为最小值的,同时Set存的是pair类型,第一关键字是:权值,val,第二关键字为:历史编号 ,NoSet直接找到对应的历史编号进行标记。同时在Set删除,然后对于队列来说,重新Push进去一个val值。

附上代码:

 1 #include<bits/stdc++.h>
 2 using namespace std;
 3 const int N = 3e5+100;
 4 queue <int> Q ;
 5 set< pair<int,int> > S;
 6 int cur ,val[N] , top ;
 7 bool deleted[N];
 8 void push( int v ){
 9     cur ++ ;
10     val[cur] = v;
11     Q.push(cur);
12     S.insert( make_pair(v,cur) );
13 }
14 int pop(){
15     if( S.empty() ) return -1;
16     while( deleted[Q.front()] ){
17         Q.pop();
18     }
19     int ret = Q.front();
20     Q.pop();
21     S.erase( make_pair(val[ret],ret) );
22     return val[ret];
23 }
24 void mov(){
25     if( S.empty() ) return ;
26     deleted[S.begin()->second] = true ;
27     push(val[S.begin()->second]);
28     S.erase( S.begin() );
29 }
30 int main()
31 {
32     int n,opt,v;
33     scanf("%d",&n);
34     while(n--){
35         scanf("%d",&opt);
36         if( opt == 1 ){
37             scanf("%d",&v);
38             push(v);
39         }else if( opt == 2 ){
40             printf("%d\n",pop());
41         }else{
42             mov();
43         }
44     }
45     return 0;
46 }

维护队列

 

原文地址:https://www.cnblogs.com/Osea/p/10995068.html

时间: 2024-10-19 00:15:13

【数据结构】维护队列的相关文章

BZOJ2453: 维护队列

2453: 维护队列 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 183  Solved: 89[Submit][Status] Description 你小时候玩过弹珠吗? 小朋友A有一些弹珠,A喜欢把它们排成队列,从左到右编号为1到N.为了整个队列鲜艳美观,小朋友想知道某一段连续弹珠中,不同颜色的弹珠有多少.当然,A有时候会依据个人喜好,替换队列中某个弹珠的颜色.但是A还没有学过编程,且觉得头脑风暴太浪费脑力了,所以向你来寻求帮助. Inp

bzoj 2453: 维护队列

2453: 维护队列 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1079  Solved: 503[Submit][Status][Discuss] Description 你小时候玩过弹珠吗? 小朋友A有一些弹珠,A喜欢把它们排成队列,从左到右编号为1到N.为了整个队列鲜艳美观,小朋友想知道某一段连续弹珠中,不同颜色的弹珠有多少.当然,A有时候会依据个人喜好,替换队列中某个弹珠的颜色.但是A还没有学过编程,且觉得头脑风暴太浪费脑力了,所以向

bzoj 2453 : 维护队列 带修莫队

2453: 维护队列 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 952  Solved: 432[Submit][Status][Discuss] Description 你小时候玩过弹珠吗? 小朋友A有一些弹珠,A喜欢把它们排成队列,从左到右编号为1到N.为了整个队列鲜艳美观,小朋友想知道某一段连续弹珠中,不同颜色的弹珠有多少.当然,A有时候会依据个人喜好,替换队列中某个弹珠的颜色.但是A还没有学过编程,且觉得头脑风暴太浪费脑力了,所以向你

Bzoj 2453: 维护队列 &amp;&amp; Bzoj 2120: 数颜色 分块,bitset

2453: 维护队列 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 578  Solved: 247[Submit][Status][Discuss] Description 你小时候玩过弹珠吗? 小朋友A有一些弹珠,A喜欢把它们排成队列,从左到右编号为1到N.为了整个队列鲜艳美观,小朋友想知道某一段连续弹珠中,不同颜色的弹珠有多少.当然,A有时候会依据个人喜好,替换队列中某个弹珠的颜色.但是A还没有学过编程,且觉得头脑风暴太浪费脑力了,所以向你

一些常用的数据结构维护手法

这篇会理论上讲一讲常用的数据结构维护手法. 我是嘴巴选手我自豪! ①cdq分治 现在我们有一些修改,有一些询问,修改之间独立. 我们考虑分治,对于左右两半分别分治,然后对于左边的修改计算对右边询问的贡献. 本身的复杂度是O(nlogn). ②整体二分 现在我们有一些修改,有一些询问. 我们需要求出,在最少多少组修改之后满足题目条件.(或者可以转化成这样) 对于单组询问,我会二分!对于多组询问,真不巧,二分超时了... 我们考虑整体二分.整体二分的框架大概是这样: def 整体二分(el,er,q

简单数据结构之队列模拟

1 /************************************************************************************** 2 * Function : 模拟队列 3 * Create Date : 2014/04/23 4 * Author : NTSK13 5 * Email : [email protected] 6 * Copyright : 欢迎大家和我一起交流学习,转载请保持源文件的完整性. 7 * 任何单位和个人不经本人允许不

数据结构 - 链队列的实行(C语言)

数据结构-链队列的实现 1 链队列的定义 队列的链式存储结构,其实就是线性表的单链表,只不过它只能尾进头出而已, 我们把它简称为链队列.为了操作上的方便,我们将队头指针指向链队列的头结点,而队尾指针指向终端结点,如下图所示. 空队列时,front和rear都指向头结点,如下图所示. 链队列的结构为: typedef int QElemType; /* QElemType类型根据实际情况而定,这里假设为int */ typedef struct QNode /* 结点结构 */ { QElemTy

数据结构 - 顺序队列的实行(C语言)

数据结构-顺序队列的实现 1 顺序队列的定义 线性表有顺序存储和链式存储,队列作为一种特殊的线性表,也同样存在这两种存储方式.我们先来看队列的顺序存储结构. 队列的顺序储存结构:用数组存储队列,为了避免当只有一个元素时,队头和队尾重合使得处理变得麻烦,所以引入两个指针:front指针指向队头元素,rear指针指向队尾元素的下一个位置,当front=rear时,为空队列,结构如下图所示. 假设是长度为5的数组,初始状态,空队列如下图左所示,front与 rear指针均指向下标为0的位置.然后入队a

[bzoj2453]维护队列_带修改莫队

维护队列 bzoj-2453 题目大意:给定一个n个数序列,支持查询区间数的种类数,单点修改.不强制在线. 注释:$1\le n,m\le 10^5$. 想法: 带修改莫队裸题. 如果没有修改操作的话,我们就正常按照莫队一样左右移动区间即可. 有了修改操作的话,我们把块变成$n^{\frac{2}{3}}$,关键字变成:左端点所在块.右端点所在块和时间戳. 然后暴力就行了. Code: #include <iostream> #include <cstdio> #include &