一、关于优先队列

队列(queue)这种东西广大OIer应该都不陌生,或者说,队列都不会你还学个卵啊(╯‵□′)╯︵┻━┻咳咳,通俗讲,队列是一种只允许从前端(队头)删除元素、从后端(队尾)插入元素的数据结构。而优先队列(priority queue)是一种赋予每个队列中元素以一个优先级的队列。在执行删除操作时,优先队列会删除具有最高优先级的元素。如此奇妙的优先队列有什么用呢,举个例子,给定一个长为n的序列和m组询问,对于每组询问,我们要找出删去序列中最小的数,再向序列加入一个数。朴素的想法是对每个询问从头到尾扫一遍,找出最小值,时间复杂度为O(nm)。而优先队列可以将时间复杂度降低到O(mlgn)的级别。那它是怎么做到的呢?其实,优先队列(又称堆)是一颗完全二叉树,其每个子节点与父节点间都具有某种特性(即为我们规定的优先级)。二叉树的性质决定了优先队列维护操作O(lgn)的复杂度。

二、STL之priority_queue

优先队列好用归好用,然而,它不好写_(:з」∠)_但是,STL(Standard Template Library 标准模板库)里给出了优先队列的模板~\(≧▽≦)/~

priority_queue定义在头文件<queue>里,和普通队列一样,priority_queue拥有如下操作:

1、priority_queue<Type, Container, Functional>Q:创建一个新的优先队列Q,其Type为其数据类型,Container 为保存数据的容器,Functional 为元素比较方式;

  特别说明:Container必须是用数组实现的容器,默认为vector;Functional默认为operator <。所以如果我们把Container和Functional都缺省,优先队列就是一个队头元素最大的大根堆。

2、Q.top():返回队头元素;

3、Q.pop():删除队列头部元素;

4、Q.push(x):在队尾增加元素x;

5、Q.empty():判断队列是否为空(如果队列为空则返回true,否则,返回false);

6、Q.size():返回队列中元素个数。

三、OpenJudge-4980拯救行动题解

描述

公主被恶人抓走,被关押在牢房的某个地方。牢房用N*M (N, M <= 200)的矩阵来表示。矩阵中的每项可以代表道路(@)、墙壁(#)、和守卫(x)。 
英勇的骑士(r)决定孤身一人去拯救公主(a)。我们假设拯救成功的表示是“骑士到达了公主所在的位置”。由于在通往公主所在位置的道路中可能遇到守卫,骑士一旦遇到守卫,必须杀死守卫才能继续前进。 
现假设骑士可以向上、下、左、右四个方向移动,每移动一个位置需要1个单位时间,杀死一个守卫需要花费额外的1个单位时间。同时假设骑士足够强壮,有能力杀死所有的守卫。

给定牢房矩阵,公主、骑士和守卫在矩阵中的位置,请你计算拯救行动成功需要花费最短时间。

输入

第一行为一个整数S,表示输入的数据的组数(多组输入)
随后有S组数据,每组数据按如下格式输入 
1、两个整数代表N和M, (N, M <= 200). 
2、随后N行,每行有M个字符。"@"代表道路,"a"代表公主,"r"代表骑士,"x"代表守卫, "#"代表墙壁。

输出

如果拯救行动成功,输出一个整数,表示行动的最短时间。
如果不可能成功,输出"Impossible"

原题链接→_→OpenJudge-4980拯救行动

读了题我们发现,这就是一个宽搜嘛。我们只要以时间定义优先级,把每个点能拓展的状态放入优先队列处理就可以了。

愉快的贴上代码:

  1. #include<cstdio>
  2. #include<queue>
  3. #include<cstring>
  4. using namespace std;
  5. const int MAXL=;
  6. int dir[]={-,,,},dir_[]={,,-,};
  7. int s;
  8. int n,m;
  9. int map[MAXL][MAXL];
  10. bool vis[MAXL][MAXL];
  11. int dx,dy;
  12. struct node
  13. {
  14. int x,y,time;
  15. friend bool operator < (node A,node B){return A.time>B.time;}
  16. //重载 < ,定义时间为优先级,时间小的元素置于队首
  17. };
  18. priority_queue <node> q;//priority_queue <node,vector<node>,operator <> q;
  19. int main()
  20. {
  21. scanf("%d",&s);
  22. while(s--)
  23. {
  24. while(!q.empty())q.pop();//清空队列q,注意:没有q.clear()的用法
  25. memset(vis,false,sizeof(vis));
  26. memset(map,,sizeof(map));
  27. scanf("%d%d",&n,&m);
  28. for(int i=;i<=n;++i)
  29. {
  30. char c[MAXL];
  31. scanf("%s",c);
  32. for(int j=;j<m;++j)
  33. {
  34. if(c[j]=='a')dx=i,dy=j+;
  35. else if(c[j]=='r')q.push((node){i,j+,}),vis[i][j+]=true;
  36. else if(c[j]=='x')map[i][j+]=;//1代表守卫
  37. else if(c[j]=='#')map[i][j+]=;//2代表墙壁
  38. }
  39. }
  40. while(!q.empty())
  41. {
  42. int x=q.top().x,y=q.top().y,time=q.top().time;
  43. if(x==dx&&y==dy)break;
  44. for(int i=;i<;++i)
  45. {
  46. if(x+dir[i]<||x+dir[i]>n||y+dir_[i]<||y+dir_[i]>m)continue;
  47. if(vis[x+dir[i]][y+dir_[i]]||map[x+dir[i]][y+dir_[i]]==)continue;
  48. vis[x+dir[i]][y+dir_[i]]=true;
  49. if(map[x+dir[i]][y+dir_[i]]==)q.push((node){x+dir[i],y+dir_[i],time+});
  50. else if(map[x+dir[i]][y+dir_[i]]==)q.push((node){x+dir[i],y+dir_[i],time+});
  51. }
  52. q.pop();
  53. }
  54. if(q.empty())printf("Impossible");
  55. else printf("%d",q.top().time);
  56. printf("\n");
  57. }
  58. return ;
  59. }

OpenJudge-4980拯救行动

弱弱地说一句,本蒟蒻码字也不容易,转载请注明出处http://www.cnblogs.com/Maki-Nishikino/p/6056072.html

【STL】优先队列priority_queue详解+OpenJudge-4980拯救行动的更多相关文章

  1. C++ STL 优先队列 priority_queue 详解(转)

    转自https://blog.csdn.net/c20182030/article/details/70757660,感谢大佬. 优先队列 引入 优先队列是一种特殊的队列,在学习堆排序的时候就有所了解 ...

  2. 优先队列priority_queue详解

    转载链接

  3. STL bind1st bind2nd详解

    STL bind1st bind2nd详解   先不要被吓到,其实这两个配接器很简单.首先,他们都在头文件<functional>中定义.其次,bind就是绑定的意思,而1st就代表fir ...

  4. C++ STL bitset 容器详解

    C++ STL bitset 容器详解 本篇随笔讲解\(C++STL\)中\(bitset\)容器的用法及常见使用技巧. \(bitset\)容器概论 \(bitset\)容器其实就是个\(01\)串 ...

  5. 2.3 C++STL vector容器详解

    文章目录 2.3.1 引入 2.3.2 代码实例 2.3.3 运行结果 总结 2.3.1 引入 vector 容器 动态数组 可变数组 vector容器 单口容器(尾部操作效率高) vector动态增 ...

  6. multimap 和priority_queue详解

    上一期是关于STL和并查集结合的例题,也附了STL中部分容器的使用摘要,由于是从网上东拼西凑的,感觉有的关键点还是没解释清楚,现在从其中摘出两个容器,用例题对它们的用法进行进一步解释. 以下是例题的介 ...

  7. [GeekBand] STL 仿函数入门详解

    本文参考文献::GeekBand课堂内容,授课老师:张文杰 :C++ Primer 11 中文版(第五版) page 37 :网络资料: 叶卡同学的部落格  http://www.leavesite. ...

  8. C++ STL 优先队列 (priority_queue)

    std::priority_queue <queue> 优先队列   优先队列是一种容器适配器,根据某些严格的弱排序标准,使其第一个元素始终包含的最大元素.   这种特性类似于堆,它可以在 ...

  9. C++ STL之vector详解

    转自http://blog.sina.com.cn/s/blog_9f1c0931010180cy.html Vectors   vector是C++标准模板库中的部分内容,它是一个多功能的,能够操作 ...

随机推荐

  1. 数据结构作业之用队列实现的基数排序(Java版)

    题目: 利用队列实现对某一个数据序列的排序(采用基数排序),其中对数据序列的数据(第1和第2条进行说明)和队列的存储方式(第3条进行说明)有如下的要求: 1)当数据序列是整数类型的数据的时候,数据序列 ...

  2. [原创]如何在Parcelable中使用泛型

    [原创]如何在Parcelable中使用泛型 实体类在实现Parcelable接口时,除了要实现它的几个方法之外,还另外要定义一个静态常量CREATOR,如下例所示: public static cl ...

  3. find命令的拾遗

    find -maxdepth 1 -regextype posix-extended -type f -regex "\./${name}[-\.].*\.(gz|bz2|tgz|zip|t ...

  4. [SharePoint 2007/2010]Query SharePoint Calendar Event

    首先要搞清楚日历事件的各种类型,参考文章: http://sharepoint.microsoft.com/blog/Pages/BlogPost.aspx?PageType=4&ListId ...

  5. hadoop 性能调优与运维

    hadoop 性能调优与运维 . 硬件选择 . 操作系统调优与jvm调优 . hadoop运维 硬件选择 1) hadoop运行环境 2)  原则一: 主节点可靠性要好于从节点 原则二:多路多核,高频 ...

  6. mogodb监控脚本

    mongodb_server.py #! /bin/env python #-*- coding:utf8 -*- import sys import os from bson.timestamp i ...

  7. 通过代码自定义cell(cell的高度不一致)

  8. 网页3D引擎“Babylon.JS”入门教程翻译总结

    使用三个月的业余时间把官方教程的入门部分译为中文并上传到github,在下一步编程前做一个总结. 历程: 最早接触游戏编程是在大三下学期,用汇编语言和实验室里的单片机.触摸屏.电机(提供声效)编的打地 ...

  9. leetcode pow(x,n)实现

    题目描述: 自己实现pow(double x, int n)方法 实现思路: 考虑位运算.考虑n的二进制表示形式,以n=51(110011)为例,x^51 = x^1*x^2*x^16*x^32,因此 ...

  10. python web前端

    概述 HTML是英文Hyper Text Mark-up Language(超文本标记语言)的缩写,他是一种制作万维网页面标准语言(标记).相当于定义统一的一套规则,大家都来遵守他,这样就可以让浏览器 ...