一、priority_queue的介紹
-
優(yōu)先級隊列是一種容器適配器,根據(jù)嚴格的弱排序標準,它的第一個元素總是它所包含的元素中最大的。
-
此上下文類似于堆,在堆中可以隨時插入元素,并且只能檢索最大堆元素(優(yōu)先級隊列中位于頂部的元素)。
-
優(yōu)先級隊列被實現(xiàn)為容器適配器,容器適配器即將特定容器類封裝作為其底層容器類,queue 提供一組特定的成員函數(shù)來訪問其元素。元素從特定容器的“尾部”彈出,其稱為優(yōu)先級隊列的頂部。
-
底層容器可以是任何標準容器類模板,也可以是其他特定設(shè)計的容器類。容器應(yīng)該可以通過隨機訪問、迭代器訪問,并支持以下操作:
-
empty():檢測容器是否為空
-
size():返回容器中有效元素的個數(shù)
-
front():返回容器中第一個元素的引用
-
push_back():在容器尾部插入元素
-
pop_back():刪除容器尾部元素
-
-
標準容器類 vector 和 deque 滿足這些需求。默認情況下,如果沒有為特定的 priority_queue 類實例化指定容器類,則使用 vector。
-
需要支持隨機訪問迭代器,以便始終在內(nèi)部保持堆結(jié)構(gòu)。容器適配器通過在需要時自動調(diào)用算法函數(shù) make_heap、push_heap 和 pop_heap 來自動完成次操作。
二、priority_queue的使用
優(yōu)先級隊列默認使用 vector 作為其底層存儲數(shù)據(jù)的容器,在 vector 上又使用了堆算法將 vector 中元素構(gòu)造成堆的結(jié)構(gòu),因此 priority_queue 就是堆,所有需要用到堆的位置,都可以考慮使用 priority_queue。注意:默認情況下 priority_queue 是大堆。
函數(shù)聲明 | 接口說明 |
---|---|
priority_queue() | 構(gòu)造一個空的優(yōu)先級隊列 |
empty() | 檢測優(yōu)先級隊列是否為空,是返回 true,否則返回 false |
top() | 返回優(yōu)先級隊列中最大(最?。┰兀炊秧斣?/td> |
push(x) | 在優(yōu)先級隊列中插入元素 x |
pop() | 刪除優(yōu)先級隊列中最大(最下)元素,即堆頂元素 |
小Tips:默認情況下,priority_queue 是大堆(默認第三個模板參數(shù)是 less);如果在 priority_queue 中放自定義類型的數(shù)據(jù),用戶需要在自定義類型中提供 > 或者 < 的重載。
2.1 數(shù)組中的第k個最大元素
class Solution {
public:
int findKthLargest(vector<int>& nums, int k)
{
//建一個大堆,向下調(diào)整建堆的時間復(fù)雜度是O(N)
priority_queue<int> pq(nums.begin(), nums.end());
//pop k-1次,時間復(fù)雜度是O(K*logN)
while(--k)
{
pq.pop();
}
return pq.top();
}
};
上面這種做法當(dāng) K 的值接近 N 的時候,它的時間復(fù)雜度就是 O(N*logN),是不滿足題目要求的,下面再用另外一種方法來解決。
lass Solution {
public:
int findKthLargest(vector<int>& nums, int k)
{
//用前k個數(shù)建一個小堆
priority_queue<int, vector<int>, greater<int>> pq(nums.begin(), nums.begin() + k);
//遍歷剩下的 N-k 個,比堆頂大就換進去
//時間復(fù)雜度是 (N-K)logN
for(size_t i = k; i < nums.size(); i++)
{
if(nums[i] > pq.top())
{
pq.pop();
pq.push(nums[i]);
}
}
return pq.top();
}
};
上面這種解法是先用數(shù)組中的前 K 個元素建一個小堆,然后從數(shù)組的第 K+1 個元素開始往后遍歷,遇到比堆頂元素大的就將堆頂?shù)脑?pop 出來,將當(dāng)前元素 push 進去。建堆的時間復(fù)雜度是 O(K),更新小堆的時間復(fù)雜度是 O((N-K)logK),這種做法當(dāng) K 很小的時候時間復(fù)雜度可以近似看做 O(NlogK),當(dāng) K 很大的時候,時間復(fù)雜度可以近似看做 O(logK)。
三、priority_queue模擬實現(xiàn)
3.1 仿函數(shù)
仿函數(shù)本質(zhì)上一個類,之所以把它叫仿函數(shù)是因為這個類的對象可以像函數(shù)一樣去使用。舉例如下:
//一個仿函數(shù)
template<class T>
class Less
{
public:
bool operator()(const T& x, const T& y)
{
return x < y;
}
};
int main()
{
Less<int> lessfunc;//定義一個對象
cout << lessfunc(1, 2) << endl;//該類型對象可以像函數(shù)一樣去使用
//cout << lessfunc.operator()(1, 2) << endl;//和上面等價
return 0;
}
小Tips:仿函數(shù)一般都是類模板,這樣就可以支持不同類型的大小比較,前提是該種類型重載實現(xiàn)了比較運算符。仿函數(shù)的誕生是為了代替函數(shù)指針,函數(shù)指針的可讀性比較差。
3.2 成員變量
template<class T, class Container=std::vector<T>, class Compare = Less<T>>
class priority_queue
{
public:
//成員
private:
Container _con;
};
小Tips:需要注意這里有三個模板參數(shù),第一個模板參數(shù)表示要在優(yōu)先級隊列中存儲的數(shù)據(jù)類型;優(yōu)先級隊列本質(zhì)上也是一個容器適配器,所以第二個模板參數(shù)表示優(yōu)先級隊列要使用的底層容器;第三個模板參數(shù)是一個仿函數(shù),用來進行大小比較的,因為優(yōu)先級隊列中會涉及到建大堆還是建小堆,中間會涉及到比較,如果沒有這個仿函數(shù),那么大小比較就只能寫死了,這樣不太好。
3.3 成員函數(shù)
3.3.1 構(gòu)造函數(shù)
priority_queue()
{}
template<class InputeIterator>
priority_queue(InputeIterator first, InputeIterator last)
{
//先將數(shù)據(jù)插入
while (first != last)
{
_con.push_back(*first);
++first;
}
//建堆,從最后一個非葉子節(jié)點開始向下調(diào)整
for (int i = (_con.size() - 1) / 2; i >= 0; i--)
{
AdjustDown(i);
}
}
小Tips:迭代器區(qū)間構(gòu)造函數(shù)是一個函數(shù)模板,只要是能支持 ++ 操作的迭代器區(qū)間都可以,即單向迭代器、雙向迭代器、隨機迭代都可以。其次將數(shù)據(jù)插入容器后還需要建堆,這里采用向下調(diào)整建堆,它的時間復(fù)雜度是 O(N)。
3.3.2 AdjustDown
void AdjustDown(int parent)
{
Compare com;
while (parent * 2 + 1 < (int)_con.size())
{
//找到最大的孩子
int maxchild = parent * 2 + 1;
if (maxchild + 1 < (int)_con.size() && com(_con[maxchild], _con[maxchild + 1]))
{
maxchild++;
}
//和父節(jié)點比較
if (com(_con[parent], _con[maxchild]))
{
std::swap(_con[maxchild], _con[parent]);
parent = maxchild;
}
else
{
break;
}
}
}
小Tips:在仿函數(shù)的加持下,向下調(diào)整代碼中的大小比較不再固定,建大堆和小堆這一份代碼就夠了,最終是大堆還是小堆是由仿函數(shù)來控制的。
3.3.3 push
void push(const T& val)
{
_con.push_back(val);
AdjustUp(_con.size() - 1);
}
小Tips:在優(yōu)先級隊列的尾部追加數(shù)據(jù),會涉及到向上調(diào)整,向上調(diào)整的代碼如下所示。
3.3.4 AdjustUp
void AdjustUp(int child)
{
Compare com;
int parent = (child - 1) / 2;
while (child > 0)//父親不會小于0,因此這里的判斷條件要用孩子大于0,孩子等于0說明已經(jīng)調(diào)整到根節(jié)點,就無需繼續(xù)調(diào)整了
{
if (com(_con[parent], _con[child]))
{
std::swap(_con[child], _con[parent]);
child = parent;
parent = (child - 1) / 2;//這里parent不會小于0
}
else
{
break;
}
}
}
3.3.5 pop
void pop()
{
std::swap(_con[0], _con[_con.size() - 1]);
_con.pop_back();
AdjustDown(0);
}
小Tips:優(yōu)先級隊列中出數(shù)據(jù),出的是堆頂?shù)臄?shù)據(jù),堆頂?shù)臄?shù)據(jù)也就是容器中的第一個數(shù)據(jù),如果底層容器是 vector 那么堆頂?shù)臄?shù)據(jù)就是下標為 0 的數(shù)據(jù),出堆頂?shù)臄?shù)據(jù)不能直接使用頭刪,這樣會導(dǎo)致后面數(shù)據(jù)的父子關(guān)系全亂了。正確的做法是,將堆頂?shù)臄?shù)據(jù)和容器中的最后一個數(shù)據(jù)交換,然后執(zhí)行 pop_back 去刪除,最后還需要從根節(jié)點開始執(zhí)行一次向下調(diào)整,讓交換到堆頂?shù)臄?shù)據(jù)去到它應(yīng)該去的地方。
3.3.6 empty
bool empty()
{
return _con.size() == 0;
}
3.3.7 size
size_t size()
{
return _con.size();
}
四、結(jié)語
今天的分享到這里就結(jié)束啦!如果覺得文章還不錯的話,可以三連支持一下,春人的主頁還有很多有趣的文章,歡迎小伙伴們前去點評,您的支持就是春人前進的動力!文章來源:http://www.zghlxwxcb.cn/news/detail-708543.html
文章來源地址http://www.zghlxwxcb.cn/news/detail-708543.html
到了這里,關(guān)于【C++雜貨鋪】優(yōu)先級隊列的使用指南與模擬實現(xiàn)的文章就介紹完了。如果您還想了解更多內(nèi)容,請在右上角搜索TOY模板網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章,希望大家以后多多支持TOY模板網(wǎng)!