Chinaunix首页 | 论坛 | 博客
  • 博客访问: 67734
  • 博文数量: 8
  • 博客积分: 45
  • 博客等级: 民兵
  • 技术积分: 172
  • 用 户 组: 普通用户
  • 注册时间: 2012-08-13 17:01
个人简介

技术上,全力以赴成为伪全站工程师 工作上,少抱怨多干活,努力为人卖命 剩下的,实现一点点自己的想法

文章分类

全部博文(8)

文章存档

2017年(4)

2015年(1)

2014年(2)

2012年(1)

我的朋友

分类: C/C++

2015-03-18 16:15:05

priority_queue:优先级队列。她有别于以往我们理解的“先进先出”的queue,她是优先级高者先出,内部自己帮你排序,默认是堆排序。两者有何相似与不同:

相似处:
1、都只允许最底端加入元素(push),最顶端弹出元素(pop)
2、都没有迭代器,不能遍历访问(当然如果全部倒出来访问,这是防止不了的)
3、两者都支持判空(empty)和获取大小(size)的操作

不同之处:
       1、priority_queue只允许访问最顶端元素(priority_queue.top), 而queue可以访问首尾queue.front和queue.back.
2、priority_queue内部采取的最大堆的算法,每次弹出的元素都是优先级最高的元素,并且加入或弹出元素,内部元素都需要重新调整结构,说白了就是排序;
     顺便唠叨一下堆排序:特性:不稳定的排序,时间复杂度为O(nlogn),并且对原始记录的排序状态不敏感,无论最好,最坏和平均,时间复杂度都是O(nlogn)。
通常在对平均时间要求严格,数据量大,并且出现最坏情况的可能性大的场景,可以考虑使用堆排序。
3、由于需要计算优先级,所以如果不是基本数据类型,则必须重载operator

要会用,还得知道她有哪些API

点击(此处)折叠或打开

  1. priority_queue(InputIterator frist, InputIterator last) /*基本数据类型使用*/
  2. priority_queue(InputIterator frist, InputIterator last, const Compare &x)/*复杂数据类型,需要提供比较函数*/
  3. /*补充一点:基本数据类型的指针可以作为InputIterator传入,例子如下:
  4. int ia[9] = {0,1,2,3,4,5,6,7,8};
  5. priority_queue ipq(ia, ia+9);
  6. */
  7. bool empty()/*判空*/
  8. size_type size()/*队列大小*/
  9. const_reference top()/*获取顶端元素,注意这里不会弹出*/
  10. void push(const value_type &x)/*增加元素*/
  11. void pop()/*弹出元素*/
什么情况下,我们会需要用到priority_queue?来看个实际的例子:
题目来源:

Merge k Sorted Lists

 
Merge k sorted linked lists and return it as one sorted list. Analyze and describe its complexity.

点击(此处)折叠或打开

  1. /**
  2.  * Definition for singly-linked list.
  3.  * struct ListNode {
  4.  * int val;
  5.  * ListNode *next;
  6.  * ListNode(int x) : val(x), next(NULL) {}
  7.  * };
  8.  */
  9. class Solution {
  10. private:
  11.     struct cmp
  12.     {
  13.            /*题目已经定义ListNode的结构,并且没有重载<,所以只能采取这种方式了*/
  14.         bool operator ()(const ListNode *node1, const ListNode *node2)
  15.         {
  16.                 return node1->val>node2->val;
  17.         }
  18.     };
  19.     
  20. public:
  21.     ListNode *mergeKLists(vector<ListNode *> &lists) {
  22.         int size = lists.size();

  23.         ListNode head(0);
  24.         ListNode *res = &head;
  25.         ListNode *tmp = NULL;
  26.         priority_queue<ListNode*, vector<ListNode*>, cmp> que;        
  27.           . /*priority_queue中的三个参数,后两个可以省去,因为有默认参数,但是如果有第三个参数的话,必定要写第二个参数*/
  28.         if(size == 0)
  29.             return NULL;
  30.             
  31.         for(int i=0; i<size; ++i)
  32.             if(lists[i])
  33.                 que.push(lists[i]);
  34.                 
  35.         while(!que.empty())
  36.         {
  37.             tmp = que.top();
  38.             que.pop();
  39.             res->next = tmp;
  40.             res = tmp;
  41.              
  42.             if(tmp->next)
  43.                 que.push(tmp->next);
  44.         }
  45.         return head.next;
  46.     }
  47. };

Note:priority_queue通常不被认为是容器,而认为是容器适配器.底部容器可以是vector、list等容器,本文没有详细讨论。在这个实例里,我是看上了他的堆排序,呜呜~~~

STL总是给人一种站在巨人肩上的感觉,爽

阅读(3453) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~