C++中STL的優(yōu)先隊(duì)列priority_queue詳解
淺談C++ STL中的優(yōu)先隊(duì)列(priority_queue)
首先函數(shù)在頭文件中,歸屬于命名空間std,使用的時(shí)候需要注意。
隊(duì)列有兩種常用的聲明方式:
std::priority_queue<T> pq; std::priority_queue<T, std::vector<T>, cmp> pq;
第一種實(shí)現(xiàn)方式較為常用,接下來我給出STL中的對(duì)應(yīng)聲明,再加以解釋。
template<class _Ty, class _Container = vector<_Ty>, class _Pr = less<typename _Container::value_type> > class priority_queue
大家可以看到,默認(rèn)模板有三個(gè)參數(shù),第一個(gè)是優(yōu)先隊(duì)列處理的類,第二個(gè)參數(shù)比較有特點(diǎn),是容納優(yōu)先隊(duì)列的容器。
實(shí)際上,優(yōu)先隊(duì)列是由這個(gè)容器+C語言中關(guān)于heap的相關(guān)操作實(shí)現(xiàn)的。
這個(gè)容器默認(rèn)是vector,也可以是dequeue,因?yàn)楹笳吖δ芨鼜?qiáng)大,而性能相對(duì)于vector較差,考慮到包裝在優(yōu)先隊(duì)列后,后者功能并不能很好發(fā)揮,所以一般選擇vector來做這個(gè)容器。
第三個(gè)參數(shù)比較重要,支持一個(gè)比較結(jié)構(gòu),默認(rèn)是less,默認(rèn)情況下,會(huì)選擇第一個(gè)參數(shù)決定的類的<運(yùn)算符來做這個(gè)比較函數(shù)。
接下來開始坑爹了,雖然用的是less結(jié)構(gòu),然而,隊(duì)列的出隊(duì)順序卻是greater的先出!
就是說,這里這個(gè)參數(shù)其實(shí)很傲嬌,表示的意思是如果!cmp,則先出列,不管這樣實(shí)現(xiàn)的目的是啥,大家只能接受這個(gè)實(shí)現(xiàn)。
實(shí)際上,這里的第三個(gè)參數(shù)可以更換成greater,像下面這樣:
std::priority_queue<T, std::vector<T>, greater<T>> pq;
一般大家如果是自定義類就干脆重載<號(hào)時(shí)注意下方向了,沒人在這里麻煩,這個(gè)選擇基本上是在使用int類還想小值先出列時(shí)。
從上面的剖析我們也就知道了,想要讓自定義類能夠使用優(yōu)先隊(duì)列,我們要重載小于號(hào)。
class Student { int id; char name[20]; bool gender; bool operator < (Student &a) const { return id > a.id; } };
就拿這個(gè)例子說,我們想讓id小的先出列,怎么辦,就要很違和的給這個(gè)小于符號(hào)重載成實(shí)際上是大于的定義。
如果我們不使用自定義類,又要用非默認(rèn)方法去排序怎么辦?
就比如說在Dijkstra中,我們當(dāng)然不會(huì)用點(diǎn)的序號(hào)去排列,無論是正序還是反序,我們想用點(diǎn)到起點(diǎn)的距離這個(gè)值來進(jìn)行排序,我們?cè)鯓幼瞿兀?/p>
優(yōu)先隊(duì)列默認(rèn)使用的是小于結(jié)構(gòu),而上文的做法是為我們的自定義類去定義新的小于結(jié)構(gòu)來符合優(yōu)先隊(duì)列,我們當(dāng)然也可以自定義比較結(jié)構(gòu)。
自定義方法以及使用如下,我直接用Dijkstra代碼來說明:
int cost[MAX_V][MAX_V]; int d[MAX_V], V, s; //自定義優(yōu)先隊(duì)列l(wèi)ess比較函數(shù) struct cmp { bool operator()(int &a, int &b) const { //因?yàn)閮?yōu)先出列判定為!cmp,所以反向定義實(shí)現(xiàn)最小值優(yōu)先 return d[a] > d[b]; } }; void Dijkstra() { std::priority_queue<int, std::vector<int>, cmp> pq; pq.push(s); d[s] = 0; while (!pq.empty()) { int tmp = pq.top();pq.pop(); for (int i = 0;i < V;++i) { if (d[i] > d[tmp] + cost[tmp][i]) { d[i] = d[tmp] + cost[tmp][i]; pq.push(i); } } } }
優(yōu)先隊(duì)列的日常使用,了解上面那些就已經(jīng)足夠。下面給出優(yōu)先隊(duì)列的所有成員函數(shù)的STL實(shí)現(xiàn)方法,希望你看完沒有一臉臥槽的感覺。c就是你聲明時(shí)候的那個(gè)vector或者其他容器。
void push(value_type&& _Val) { // insert element at beginning c.push_back(_STD move(_Val)); push_heap(c.begin(), c.end(), comp); } template<class... _Valty> void emplace(_Valty&&... _Val) { // insert element at beginning c.emplace_back(_STD forward<_Valty>(_Val)...); push_heap(c.begin(), c.end(), comp); } bool empty() const { // test if queue is empty return (c.empty()); } size_type size() const { // return length of queue return (c.size()); } const_reference top() const { // return highest-priority element return (c.front()); } void push(const value_type& _Val) { // insert value in priority order c.push_back(_Val); push_heap(c.begin(), c.end(), comp); } void pop() { // erase highest-priority element pop_heap(c.begin(), c.end(), comp); c.pop_back(); }
到此這篇關(guān)于C++中STL的優(yōu)先隊(duì)列priority_queue詳解的文章就介紹到這了,更多相關(guān)STL的優(yōu)先隊(duì)列內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- C++ 實(shí)現(xiàn)優(yōu)先隊(duì)列的簡單實(shí)例
- c++優(yōu)先隊(duì)列(priority_queue)用法詳解
- c++優(yōu)先隊(duì)列用法知識(shí)點(diǎn)總結(jié)
- C++優(yōu)先隊(duì)列用法案例詳解
- 詳解c++優(yōu)先隊(duì)列priority_queue的用法
- C++高級(jí)數(shù)據(jù)結(jié)構(gòu)之優(yōu)先隊(duì)列
- C++實(shí)現(xiàn)優(yōu)先隊(duì)列的示例詳解
- C++示例詳解Prim算法與優(yōu)先隊(duì)列
- 深入了解C++優(yōu)先隊(duì)列(priority_queue)的使用方法
- C++優(yōu)先隊(duì)列的使用小結(jié)
- C++的實(shí)現(xiàn)優(yōu)先隊(duì)列(Priority?Queue)的實(shí)現(xiàn)
相關(guān)文章
C++設(shè)置超時(shí)時(shí)間的簡單實(shí)現(xiàn)方法
這篇文章主要介紹了C++設(shè)置超時(shí)時(shí)間的簡單實(shí)現(xiàn)方法,涉及系統(tǒng)函數(shù)setsockopt對(duì)套接口的操作,具有一定的實(shí)用價(jià)值,需要的朋友可以參考下2014-10-10C語言中數(shù)組作為函數(shù)的參數(shù)以及返回值的使用簡單入門
這篇文章主要介紹了C語言中數(shù)組作為函數(shù)的參數(shù)以及返回值的使用簡單入門,這里以一維數(shù)組作為基本條件進(jìn)行例子講解,需要的朋友可以參考下2015-12-12C++開發(fā):為什么多線程讀寫shared_ptr要加鎖的詳細(xì)介紹
本篇文章介紹了,在C++中為什么多線程讀寫shared_ptr要加鎖的詳細(xì)說明。需要的朋友參考下2013-04-04一文帶你快速了解C/C++標(biāo)準(zhǔn)庫中的ptrdiff_t
ptrdiff_t是C/C++標(biāo)準(zhǔn)庫中定義的一個(gè)與機(jī)器相關(guān)的數(shù)據(jù)類型,ptrdiff_t類型變量通常用來保存兩個(gè)指針減法操作的結(jié)果,下面這篇文章主要給大家介紹了關(guān)于C/C++標(biāo)準(zhǔn)庫中ptrdiff_t的相關(guān)資料,需要的朋友可以參考下2022-11-11