解讀堆排序算法及用C++實(shí)現(xiàn)基于最大堆的堆排序示例
1、堆排序定義
n個(gè)關(guān)鍵字序列Kl,K2,…,Kn稱(chēng)為堆,當(dāng)且僅當(dāng)該序列滿(mǎn)足如下性質(zhì)(簡(jiǎn)稱(chēng)為堆性質(zhì)):
(1) ki≤K2i且ki≤K2i+1 或(2)Ki≥K2i且ki≥K2i+1(1≤i≤ )
若將此序列所存儲(chǔ)的向量R[1..n]看做是一棵完全二叉樹(shù)的存儲(chǔ)結(jié)構(gòu),則堆實(shí)質(zhì)上是滿(mǎn)足如下性質(zhì)的完全二叉樹(shù):樹(shù)中任一非葉結(jié)點(diǎn)的關(guān)鍵字均不大于(或不小于)其左右孩子(若存在)結(jié)點(diǎn)的關(guān)鍵字。
【例】關(guān)鍵字序列(10,15,56,25,30,70)和(70,56,30,25,15,10)分別滿(mǎn)足堆性質(zhì)(1)和(2),故它們均是堆,其對(duì)應(yīng)的完全二叉樹(shù)分別如最小堆示例和最大堆示例所示。
堆排序算法
2、最大堆和最小堆
(1)根結(jié)點(diǎn)(亦稱(chēng)為堆頂)的關(guān)鍵字是堆里所有結(jié)點(diǎn)關(guān)鍵字中最小者的堆稱(chēng)為最小堆。
(2)結(jié)點(diǎn)(亦稱(chēng)為堆頂)的關(guān)鍵字是堆里所有結(jié)點(diǎn)關(guān)鍵字中最大者,稱(chēng)為最大堆。
注意:
(1)堆中任一子樹(shù)亦是堆。
(2)以上討論的堆實(shí)際上是二叉堆(Binary Heap),類(lèi)似地可定義k叉堆。
3、堆排序的基本思路如下:
(1)把待排序數(shù)組構(gòu)造成一個(gè)最大堆
(2)取出樹(shù)的根(最大(小)值, 實(shí)際算法的實(shí)現(xiàn)并不是真正的取出)
(3)將樹(shù)中剩下的元素再構(gòu)造成一個(gè)最大堆(這里的構(gòu)造和第1步不一樣,具體看實(shí)現(xiàn)部分)
(4)重復(fù)2,3操作,直到取完所有的元素
(5)把元素按取出的順序排列,即得到一個(gè)有序數(shù)組(在代碼實(shí)現(xiàn)里是通過(guò)交換操作"無(wú)形中"完成的)
在開(kāi)始實(shí)現(xiàn)算法先看幾個(gè)結(jié)論(證明略):
(1)完全二叉樹(shù)A[0:n-1]中的任意節(jié)點(diǎn),其下標(biāo)為 ii, 那么其子節(jié)點(diǎn)的下標(biāo)分別是為2i+12i+1 和 2(i+1)2(i+1)
(2)大小為n的完全二叉樹(shù)A[0:n-1],葉子節(jié)點(diǎn)中下標(biāo)最小的是⌊n2⌋⌊n2⌋, 非葉子節(jié)點(diǎn)中下標(biāo)最大的是⌊n2⌋−1⌊n2⌋−1
(3)如果數(shù)組是一個(gè)最大堆,那么最大元素就是A[0]
(4)最大堆中任意節(jié)點(diǎn)的左右子樹(shù)也是最大堆
4、實(shí)現(xiàn)示例
這里的算法實(shí)現(xiàn)使用的是最大堆,首先來(lái)解決由數(shù)組建立最大堆的問(wèn)題:
// 用于計(jì)算下標(biāo)為i的節(jié)點(diǎn)的兩個(gè)子節(jié)點(diǎn)的下標(biāo)值
#define LEFT(i) (2 * (i) + 1)
#define RIGHT(i) (2 * ((i) + 1))
/* 此函數(shù)把一顆二叉樹(shù)中以node為根的子樹(shù)變成最大堆。
* 注意: 使用的前提條件是 node節(jié)點(diǎn)的左右子樹(shù)(如果存在的話(huà))都是最大堆。
* 這個(gè)函數(shù)是整個(gè)算法的關(guān)鍵。
*/
void max_heapify(int heap[], int heap_size, int node)
{
// 這里先不考慮整數(shù)溢出的問(wèn)題
// 先把注意力放在主要的功能上
// 如果數(shù)據(jù)規(guī)模夠大,int類(lèi)型必然會(huì)溢出
int l_child = LEFT(node);
int r_child = RIGHT(node);
int max_value = node;
if (l_child < heap_size && heap[l_child] > heap[max_value])
{
max_value = l_child;
}
if (r_child < heap_size && heap[r_child] > heap[max_value])
{
max_value = r_child;
}
if (max_value != node)
{
swap_val(heap + node, heap + max_value);
// 之后還要保證被交換的子節(jié)點(diǎn)構(gòu)成的子樹(shù)仍然是最大堆
// 如果不是這個(gè)節(jié)點(diǎn)會(huì)繼續(xù)"下沉",直到合適的位置
max_heapify(heap, heap_size, max_value);
}
}
/* 將一個(gè)數(shù)組構(gòu)造成最大堆
* 自底向上的利用max_heapify函數(shù)處理
*/
void build_max_heap(int heap[], int heap_size)
{
if (heap_size < 2)
{
return;
}
int first_leaf = heap_size >> 1;//第一個(gè)葉子節(jié)點(diǎn)的下標(biāo)
int i;
// 從最后一個(gè)非葉子節(jié)點(diǎn)開(kāi)始自底向上構(gòu)建,
// 葉子節(jié)點(diǎn)都看作最大堆,因此可以使用max_heapify函數(shù)
for (i = first_leaf - 1; i >= 0; i--)
{
max_heapify(heap, heap_size, i);
}
}
函數(shù)max_heapify將指定子樹(shù)的根節(jié)點(diǎn)"下沉"到合適的位置, 最終子樹(shù)變成最大堆, 該過(guò)程最壞時(shí)間復(fù)雜度為O(logn)O(logn)。函數(shù)build_max_heap自底向上的調(diào)用max_heapify, 最終整個(gè)數(shù)組滿(mǎn)足最大堆,迭代過(guò)程的復(fù)雜度為O(nlogn)O(nlogn), 因此整個(gè)函數(shù)的最壞時(shí)間復(fù)雜度也是O(nlogn)O(nlogn)。 而如果當(dāng)前數(shù)組已經(jīng)是最大堆了,例如數(shù)組原本是降序排列的, 那么max_heapify過(guò)程的時(shí)間復(fù)雜度就是O(1)O(1), 此時(shí)build_max_heap的時(shí)間復(fù)雜度是O(n)O(n),這是最好的情況。
接著實(shí)現(xiàn)堆排序過(guò)程:
/* heap sort 主函數(shù)
*/
void heap_sort(int heap[], int heap_size)
{
if (heap == NULL || heap_size < 2)
{
return;
}
//構(gòu)建最大堆
build_max_heap(heap, heap_size);
int i;
for (i = heap_size - 1; i > 0; i--)
{
/* 把當(dāng)前樹(shù)的根節(jié)點(diǎn)交換到末尾
* 相當(dāng)于取出最大值,樹(shù)的規(guī)模變小。
* 交換后的樹(shù)不是最大堆,但是根的兩顆子樹(shù)依然是最大堆
* 滿(mǎn)足調(diào)用max_heapify的條件。之所以這樣交換,
* 是因?yàn)橛胢ax_heapify處理時(shí)間復(fù)雜度較低,
* 如果不交換而直接"取出"heap[0], 此處可能要使用
* build_max_heap重新建立最大堆,時(shí)間復(fù)雜度較大
*/
swap_val(heap, heap + i);
heap_size--;
//維護(hù)最大堆
max_heapify(heap, heap_size, 0);
}
}
最終的堆排序算法中,build_max_heap的復(fù)雜度是已知的, 迭代部分和build_max_heap的實(shí)現(xiàn)類(lèi)似,而且不難看出, 交換后的根元素在下一次建堆過(guò)程中必然下沉到堆底,因此無(wú)論情況好壞, 該迭代過(guò)程時(shí)間復(fù)雜度都是O(nlogn)O(nlogn), 所以整個(gè)算法的最好最壞和平均時(shí)間復(fù)雜度都是O(nlogn)O(nlogn)。
堆排序算法的空間復(fù)雜度是O(1)O(1),從實(shí)現(xiàn)上很容易看出來(lái)。
上一篇:從string類(lèi)的實(shí)現(xiàn)看C++類(lèi)的四大函數(shù)(面試常見(jiàn))
欄 目:C語(yǔ)言
下一篇:淺談C++指針(必看)
本文標(biāo)題:解讀堆排序算法及用C++實(shí)現(xiàn)基于最大堆的堆排序示例
本文地址:http://www.jygsgssxh.com/a1/Cyuyan/2233.html
您可能感興趣的文章
- 04-02c語(yǔ)言編寫(xiě)函數(shù)冒泡排序 c語(yǔ)言冒泡排序法函數(shù)
- 01-10深入理解堆排序及其分析
- 01-10深入單鏈表的快速排序詳解
- 01-10內(nèi)部排序之堆排序的實(shí)現(xiàn)詳解
- 01-10用c語(yǔ)言實(shí)現(xiàn)冒泡排序,選擇排序,快速排序
- 01-10C++中靜態(tài)存儲(chǔ)區(qū)與棧以及堆的區(qū)別詳解
- 01-10C++實(shí)現(xiàn)基數(shù)排序的方法詳解
- 01-10淺析棧區(qū)和堆區(qū)內(nèi)存分配的區(qū)別
- 01-10C語(yǔ)言中堆空間的生成與釋放詳解
- 01-10歸并排序的遞歸實(shí)現(xiàn)與非遞歸實(shí)現(xiàn)代碼


閱讀排行
- 1C語(yǔ)言 while語(yǔ)句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹(shù)的示例代碼(圣誕
- 3利用C語(yǔ)言實(shí)現(xiàn)“百馬百擔(dān)”問(wèn)題方法
- 4C語(yǔ)言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語(yǔ)言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語(yǔ)言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語(yǔ)言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 04-02c語(yǔ)言函數(shù)調(diào)用后清空內(nèi)存 c語(yǔ)言調(diào)用
- 04-02func函數(shù)+在C語(yǔ)言 func函數(shù)在c語(yǔ)言中
- 04-02c語(yǔ)言的正則匹配函數(shù) c語(yǔ)言正則表達(dá)
- 04-02c語(yǔ)言用函數(shù)寫(xiě)分段 用c語(yǔ)言表示分段
- 04-02c語(yǔ)言中對(duì)數(shù)函數(shù)的表達(dá)式 c語(yǔ)言中對(duì)
- 04-02c語(yǔ)言編寫(xiě)函數(shù)冒泡排序 c語(yǔ)言冒泡排
- 04-02c語(yǔ)言沒(méi)有round函數(shù) round c語(yǔ)言
- 04-02c語(yǔ)言分段函數(shù)怎么求 用c語(yǔ)言求分段
- 04-02C語(yǔ)言中怎么打出三角函數(shù) c語(yǔ)言中怎
- 04-02c語(yǔ)言調(diào)用函數(shù)求fibo C語(yǔ)言調(diào)用函數(shù)求
隨機(jī)閱讀
- 01-10C#中split用法實(shí)例總結(jié)
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
- 01-10delphi制作wav文件的方法
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
- 04-02jquery與jsp,用jquery
- 01-11ajax實(shí)現(xiàn)頁(yè)面的局部加載
- 01-10使用C語(yǔ)言求解撲克牌的順子及n個(gè)骰子
- 01-11Mac OSX 打開(kāi)原生自帶讀寫(xiě)NTFS功能(圖文
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
- 01-10SublimeText編譯C開(kāi)發(fā)環(huán)境設(shè)置


