国产片侵犯亲女视频播放_亚洲精品二区_在线免费国产视频_欧美精品一区二区三区在线_少妇久久久_在线观看av不卡

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務器之家 - 編程語言 - C/C++ - 解讀堆排序算法及用C++實現基于最大堆的堆排序示例

解讀堆排序算法及用C++實現基于最大堆的堆排序示例

2021-04-06 13:26C++教程網 C/C++

把待排序的數組構造出最大堆是進行堆排序操作的基本方法,這里將帶大家來解讀堆排序算法及用C++實現基于最大堆的堆排序示例,首先從堆排序的概念開始:

1、堆排序定義
n個關鍵字序列Kl,K2,…,Kn稱為堆,當且僅當該序列滿足如下性質(簡稱為堆性質):
(1) ki≤K2i且ki≤K2i+1 或(2)Ki≥K2i且ki≥K2i+1(1≤i≤   )
若將此序列所存儲的向量R[1..n]看做是一棵完全二叉樹的存儲結構,則堆實質上是滿足如下性質的完全二叉樹:樹中任一非葉結點的關鍵字均不大于(或不小于)其左右孩子(若存在)結點的關鍵字。
【例】關鍵字序列(10,15,56,25,30,70)和(70,56,30,25,15,10)分別滿足堆性質(1)和(2),故它們均是堆,其對應的完全二叉樹分別如最小堆示例和最大堆示例所示。
堆排序算法

解讀堆排序算法及用C++實現基于最大堆的堆排序示例

 

2、最大堆和最小堆
(1)根結點(亦稱為堆頂)的關鍵字是堆里所有結點關鍵字中最小者的堆稱為最小堆。
(2)結點(亦稱為堆頂)的關鍵字是堆里所有結點關鍵字中最大者,稱為最大堆。
注意:
(1)堆中任一子樹亦是堆。
(2)以上討論的堆實際上是二叉堆(Binary Heap),類似地可定義k叉堆。

3、堆排序的基本思路如下:
(1)把待排序數組構造成一個最大堆
(2)取出樹的根(最大(小)值, 實際算法的實現并不是真正的取出)
(3)將樹中剩下的元素再構造成一個最大堆(這里的構造和第1步不一樣,具體看實現部分)
(4)重復2,3操作,直到取完所有的元素
(5)把元素按取出的順序排列,即得到一個有序數組(在代碼實現里是通過交換操作"無形中"完成的)
在開始實現算法先看幾個結論(證明略):
(1)完全二叉樹A[0:n-1]中的任意節點,其下標為 ii, 那么其子節點的下標分別是為2i+12i+1 和 2(i+1)2(i+1)
(2)大小為n的完全二叉樹A[0:n-1],葉子節點中下標最小的是⌊n2⌋⌊n2⌋, 非葉子節點中下標最大的是⌊n2⌋−1⌊n2⌋−1
(3)如果數組是一個最大堆,那么最大元素就是A[0]
(4)最大堆中任意節點的左右子樹也是最大堆
 
4、實現示例
這里的算法實現使用的是最大堆,首先來解決由數組建立最大堆的問題:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
// 用于計算下標為i的節點的兩個子節點的下標值
#define LEFT(i) (2 * (i) + 1)
#define RIGHT(i) (2 * ((i) + 1))
         
/* 此函數把一顆二叉樹中以node為根的子樹變成最大堆。
 * 注意: 使用的前提條件是 node節點的左右子樹(如果存在的話)都是最大堆。
 * 這個函數是整個算法的關鍵。
 */
void max_heapify(int heap[], int heap_size, int node)
{
  // 這里先不考慮整數溢出的問題
  // 先把注意力放在主要的功能上
  // 如果數據規模夠大,int類型必然會溢出
  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);
 
    // 之后還要保證被交換的子節點構成的子樹仍然是最大堆
    // 如果不是這個節點會繼續"下沉",直到合適的位置
    max_heapify(heap, heap_size, max_value);
  }
}
 
/* 將一個數組構造成最大堆
 * 自底向上的利用max_heapify函數處理
 */
void build_max_heap(int heap[], int heap_size)
{
  if (heap_size < 2)
  {
    return;
  }
  int first_leaf = heap_size >> 1;//第一個葉子節點的下標
 
  int i;
  // 從最后一個非葉子節點開始自底向上構建,
  // 葉子節點都看作最大堆,因此可以使用max_heapify函數
  for (i = first_leaf - 1; i >= 0; i--)
  {
    max_heapify(heap, heap_size, i);
  }
}

函數max_heapify將指定子樹的根節點"下沉"到合適的位置, 最終子樹變成最大堆, 該過程最壞時間復雜度為O(logn)O(log?n)。函數build_max_heap自底向上的調用max_heapify, 最終整個數組滿足最大堆,迭代過程的復雜度為O(nlogn)O(nlog?n), 因此整個函數的最壞時間復雜度也是O(nlogn)O(nlog?n)。 而如果當前數組已經是最大堆了,例如數組原本是降序排列的, 那么max_heapify過程的時間復雜度就是O(1)O(1), 此時build_max_heap的時間復雜度是O(n)O(n),這是最好的情況。

接著實現堆排序過程:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
/* heap sort 主函數
 */
void heap_sort(int heap[], int heap_size)
{
  if (heap == NULL || heap_size < 2)
  {
    return;
  }
  //構建最大堆
  build_max_heap(heap, heap_size);
 
  int i;
  for (i = heap_size - 1; i > 0; i--)
  {
    /* 把當前樹的根節點交換到末尾
     * 相當于取出最大值,樹的規模變小。
     * 交換后的樹不是最大堆,但是根的兩顆子樹依然是最大堆
     * 滿足調用max_heapify的條件。之所以這樣交換,
     * 是因為用max_heapify處理時間復雜度較低,
     * 如果不交換而直接"取出"heap[0], 此處可能要使用
     * build_max_heap重新建立最大堆,時間復雜度較大
     */
    swap_val(heap, heap + i);
 
    heap_size--;
    //維護最大堆
    max_heapify(heap, heap_size, 0);
  }
}

最終的堆排序算法中,build_max_heap的復雜度是已知的, 迭代部分和build_max_heap的實現類似,而且不難看出, 交換后的根元素在下一次建堆過程中必然下沉到堆底,因此無論情況好壞, 該迭代過程時間復雜度都是O(nlogn)O(nlog?n), 所以整個算法的最好最壞和平均時間復雜度都是O(nlogn)O(nlog?n)。
堆排序算法的空間復雜度是O(1)O(1),從實現上很容易看出來。

延伸 · 閱讀

精彩推薦
  • C/C++c++ 單線程實現同時監聽多個端口

    c++ 單線程實現同時監聽多個端口

    這篇文章主要介紹了c++ 單線程實現同時監聽多個端口的方法,幫助大家更好的理解和學習使用c++,感興趣的朋友可以了解下...

    源之緣11542021-10-27
  • C/C++C語言中炫酷的文件操作實例詳解

    C語言中炫酷的文件操作實例詳解

    內存中的數據都是暫時的,當程序結束時,它們都將丟失,為了永久性的保存大量的數據,C語言提供了對文件的操作,這篇文章主要給大家介紹了關于C語言中文件...

    針眼_6702022-01-24
  • C/C++學習C++編程的必備軟件

    學習C++編程的必備軟件

    本文給大家分享的是作者在學習使用C++進行編程的時候所用到的一些常用的軟件,這里推薦給大家...

    謝恩銘10102021-05-08
  • C/C++詳解c語言中的 strcpy和strncpy字符串函數使用

    詳解c語言中的 strcpy和strncpy字符串函數使用

    strcpy 和strcnpy函數是字符串復制函數。接下來通過本文給大家介紹c語言中的strcpy和strncpy字符串函數使用,感興趣的朋友跟隨小編要求看看吧...

    spring-go5642021-07-02
  • C/C++C語言實現電腦關機程序

    C語言實現電腦關機程序

    這篇文章主要為大家詳細介紹了C語言實現電腦關機程序,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下...

    xiaocaidayong8482021-08-20
  • C/C++深入理解goto語句的替代實現方式分析

    深入理解goto語句的替代實現方式分析

    本篇文章是對goto語句的替代實現方式進行了詳細的分析介紹,需要的朋友參考下...

    C語言教程網7342020-12-03
  • C/C++C++之重載 重定義與重寫用法詳解

    C++之重載 重定義與重寫用法詳解

    這篇文章主要介紹了C++之重載 重定義與重寫用法詳解,本篇文章通過簡要的案例,講解了該項技術的了解與使用,以下就是詳細內容,需要的朋友可以參考下...

    青山的青6062022-01-04
  • C/C++C/C++經典實例之模擬計算器示例代碼

    C/C++經典實例之模擬計算器示例代碼

    最近在看到的一個需求,本以為比較簡單,但花了不少時間,所以下面這篇文章主要給大家介紹了關于C/C++經典實例之模擬計算器的相關資料,文中通過示...

    jia150610152021-06-07
主站蜘蛛池模板: 久久99er6热线精品首页蜜臀 | 久久精品国产亚洲一区二区三区 | 国产色网 | а天堂中文最新一区二区三区 | 国产精品美女久久久久久免费 | 午夜视频网 | 免费av电影网站 | 成人av电影在线 | 成人av一区二区三区 | 白浆一区| 国产馆一区二区 | 色综合久久久 | 在线色站| 久久99国产精品久久99大师 | 日本一区二区在线观看视频 | 国产精品久久久久久吹潮 | 欧美黑人一级爽快片淫片高清 | 国产二区视频 | 国产精品一区在线观看 | 国产精品一区二区视频 | 日本精品在线观看 | 综合色九九 | 亚洲欧洲成人 | 视频精品一区 | 91中文在线 | 久久久精品欧美 | 视频在线亚洲 | 高清视频一区 | 亚洲欧美观看 | 91久久精品国产亚洲a∨麻豆 | 亚洲91| 欧美高清视频在线观看 | 成年人视频在线观看免费 | 亚洲视频一区在线播放 | 国产欧美日韩在线 | 亚洲成人久久久 | 国产精品成人一区二区三区 | 亚洲91精品 | 亚洲精品久久久蜜桃 | 亚洲日本中文字幕 | 日韩欧美国产一区二区 |