亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

C++高級數(shù)據(jù)結(jié)構(gòu)之優(yōu)先隊(duì)列

 更新時(shí)間:2022年05月24日 09:27:43   作者:下一站不是永遠(yuǎn)  
這篇文章主要介紹了C++高級數(shù)據(jù)結(jié)構(gòu)之優(yōu)先隊(duì)列,文章圍繞主題的相關(guān)資料展開詳細(xì)介紹,具有一定的參考價(jià)值,需要的小伙伴可以參考一下

前言

  • 高級數(shù)據(jù)結(jié)構(gòu)(Ⅱ)優(yōu)先隊(duì)列(Priority Queue)
  • API
  • 堆的定義
  • 二叉堆表示法
  • 堆的算法
  • 基于堆的優(yōu)先隊(duì)列
  • 堆排序

高級數(shù)據(jù)結(jié)構(gòu)(Ⅱ)優(yōu)先隊(duì)列(Priority Queue)

許多應(yīng)用程序都需要處理有序的元素,但不一定要求它們?nèi)坑行?,或是不一定要一次就將它們排序。很多情況下我們會收集一些元素,處理當(dāng)前鍵值最大的元素,然后再收集更多的元素,再處理當(dāng)前鍵值最大的元素,如此這般。例如,你可能有一臺能夠同時(shí)運(yùn)行多個(gè)應(yīng)用程序的電腦(或者手機(jī))。這是通過為每個(gè)應(yīng)用程序事件分配一個(gè)優(yōu)先級,并總是處理下一個(gè)優(yōu)先級最高的事件來實(shí)現(xiàn)的。例如,絕大多數(shù)手機(jī)分配給來電的優(yōu)先級都會被游戲程序的高。

在這種情況下,一個(gè)合適的數(shù)據(jù)結(jié)構(gòu)應(yīng)該支持兩種操作:刪除最大元素和插入元素。這種數(shù)據(jù)類型叫做優(yōu)先隊(duì)列。優(yōu)先隊(duì)列的使用和隊(duì)列(刪除最老的元素)以及棧(刪除最新的元素)類似,但高效地實(shí)現(xiàn)它則更有挑戰(zhàn)性。

在本篇中,我們會學(xué)習(xí)一種基于二叉堆數(shù)據(jù)結(jié)構(gòu)的一種優(yōu)先隊(duì)列的經(jīng)典實(shí)現(xiàn)方法,用數(shù)組保存元素并按照一定條件排序,以實(shí)現(xiàn)高效地(對數(shù)級別的)刪除最大元素和插入元素操作。

API

優(yōu)先隊(duì)列是一種抽象數(shù)據(jù)類型,它表示了一組值和對這些值的操作,它的抽象層使我們能夠方便地將應(yīng)用程序和各種具體實(shí)現(xiàn)隔離開來。

實(shí)現(xiàn)

public class MaxPQ <Key extends Comparable<Key>>
-----------------------------------------------------------------------
MaxPQ() //創(chuàng)建一個(gè)優(yōu)先隊(duì)列
MaxPQ(int max) //創(chuàng)建一個(gè)初始容量為max的優(yōu)先隊(duì)列
MaxPQ(Key[] a) //用a中的元素創(chuàng)建一個(gè)優(yōu)先隊(duì)列
void insert(Key v) //向優(yōu)先隊(duì)列中插入一個(gè)元素
Key max() //返回最大元素
Key delMax() //刪除并返回最大元素
boolean isEmpty() //返回隊(duì)列是否為空
int size() //返回優(yōu)先隊(duì)列中元素個(gè)數(shù)

堆的定義

數(shù)據(jù)結(jié)構(gòu)二叉堆能夠很好地實(shí)現(xiàn)優(yōu)先隊(duì)列的基本操作。在二叉堆的數(shù)組中,每個(gè)元素都要保證大于等于另外兩個(gè)特定位置的元素。相應(yīng)地,這些位置的元素又至少要大于等于數(shù)組中的另外兩個(gè)元素,以此類推。如果我們將所有的元素畫成一顆二叉樹,將每個(gè)較大元素和較小的元素用邊連接就可以很容易地看出這種結(jié)構(gòu)。

命題:當(dāng)一顆二叉樹的每個(gè)結(jié)點(diǎn)都大于等于它的另外兩個(gè)子節(jié)點(diǎn)時(shí),它被稱為堆有序。

相應(yīng)地,在堆有序的二叉樹中,每個(gè)結(jié)點(diǎn)都小于等于它的父節(jié)點(diǎn)(如果有的話)。從任意結(jié)點(diǎn)向上,我們都能得到一列非遞減的元素;從任意結(jié)點(diǎn)向下,我們都能得到一列非遞增的元素。

命題:根節(jié)點(diǎn)是堆有序的二叉樹中最大的結(jié)點(diǎn)

二叉堆表示法

如果我們用指針來表示堆有序的二叉樹,那么每個(gè)元素都需要三個(gè)指針來找到它的上下結(jié)點(diǎn)(父節(jié)點(diǎn)和兩個(gè)子節(jié)點(diǎn))。如下圖所示,如果我們使用完全二叉樹,表達(dá)就會變的特別方便

可以先定下根節(jié)點(diǎn),然后一層一層地由上向下、從左至右,在每個(gè)結(jié)點(diǎn)的下方連接兩個(gè)更小的結(jié)點(diǎn),直至將N個(gè)結(jié)點(diǎn)全部連接完畢。完全二叉樹只用數(shù)組而不需要指針就可以表示。具體方法就是將二叉樹的結(jié)點(diǎn)按照層級順序放入數(shù)組中,根結(jié)點(diǎn)在位置1,它的子節(jié)點(diǎn)在位置2和位置3,而子節(jié)點(diǎn)的子節(jié)點(diǎn)分別在位置4、5、6和7,以此類推。

定義:二叉堆是一組能夠用堆有序的完全二叉樹排序的元素,并在數(shù)組中按照層級存儲(不使用數(shù)組的第一個(gè)位置)

在一個(gè)堆中,位置k的結(jié)點(diǎn)的父節(jié)點(diǎn)的位置為[k/2]向下取整,而它的兩個(gè)子節(jié)點(diǎn)的位置則分別為2k和2k+1。用數(shù)組(堆)實(shí)現(xiàn)的完全二叉樹的結(jié)構(gòu)是很嚴(yán)格的,但它的靈活性已經(jīng)足以讓我們高效地實(shí)現(xiàn)優(yōu)先隊(duì)列。用它們我們能將實(shí)現(xiàn)對數(shù)級別的插入元素和刪除最大元素的操作。利用數(shù)組中無需指針即可沿樹上下移動的便利和以下性質(zhì),算法保證了對數(shù)復(fù)雜度的性能。

命題:一顆大小為N的完全二叉樹的高度為lg[N]

堆的算法

我們用長度為N+1的私有數(shù)組pq[]來表示一個(gè)大小為N的堆。我們不會使用pq[0],堆元素放在pq[1]至pq[N]中。在排序算法中,我們只通過私有輔助函數(shù)less()和exch()來訪問元素,但因?yàn)樗械脑囟荚跀?shù)組pq[]中,我們在下面基于堆的優(yōu)先隊(duì)列中會用更緊湊的實(shí)現(xiàn)方式,不再將數(shù)組作為參數(shù)傳遞。堆的操作會首先進(jìn)行一些簡單的改動,打破堆的狀態(tài),然后再遍歷堆并按照要求將堆的狀態(tài)恢復(fù)。我們稱這個(gè)過程為堆的有序化(reheapifying)。

實(shí)現(xiàn):

private boolean less(int i, int j) {
return pq[i].compareTo(pq[j]) < 0;
}
private void exch(int i, int j) {
Key t = pq[i];
pq[i] = pq[j];
pq[j] = t;
}

在堆有序化的過程中我們會遇到兩種情況。當(dāng)某個(gè)結(jié)點(diǎn)的優(yōu)先級上升(或是在堆底加入一個(gè)新的元素)時(shí),我們需要由下至上恢復(fù)堆的順序。當(dāng)某個(gè)結(jié)點(diǎn)的優(yōu)先級下降(例如,將根節(jié)點(diǎn)替換為一個(gè)較小的元素)時(shí),我們需要由下至上恢復(fù)堆的順序。首先來學(xué)習(xí)如何實(shí)現(xiàn)這兩種輔助操作,然后再用它們實(shí)現(xiàn)插入元素和刪除最大元素的操作。

由下至上的堆的有序化(上?。?/strong>

如果堆的有序狀態(tài)因?yàn)槟硞€(gè)結(jié)點(diǎn)變得比它的父節(jié)點(diǎn)更大而被打破,那么我們就需要通過交換它和它的父節(jié)點(diǎn)來修復(fù)堆。交換后,這個(gè)結(jié)點(diǎn)比它的兩個(gè)子結(jié)點(diǎn)都大(一個(gè)是曾經(jīng)的父節(jié)點(diǎn),另一個(gè)比它更小,因?yàn)樗撬?jīng)父節(jié)點(diǎn)的子節(jié)點(diǎn)),但這個(gè)結(jié)點(diǎn)仍然可能比它現(xiàn)在的父節(jié)點(diǎn)更大。我們可以一遍遍地用同樣的方法恢復(fù)秩序,將這個(gè)結(jié)點(diǎn)不斷向上移動直到我們遇到了一個(gè)更大的父節(jié)點(diǎn)。位置k的結(jié)點(diǎn)的父結(jié)點(diǎn)為[k/2],swim()方法中的循環(huán)可以保證只有位置k上的結(jié)點(diǎn)大于它的父節(jié)點(diǎn)時(shí)堆的有序狀態(tài)才會被打破。因此只要該結(jié)點(diǎn)不再大于父結(jié)點(diǎn),堆的有序狀態(tài)就恢復(fù)了。當(dāng)一個(gè)結(jié)點(diǎn)優(yōu)先級太大的時(shí)候它需要?。╯wim)到堆的更高層,詳細(xì)代碼如下

private void swim(int k) {
while(k > 1 && less(k / 2, k)) {
exch(k / 2, k);
k = k / 2;
}
}

由上至下的堆的有序化(下沉)

如果堆的有序狀態(tài)因?yàn)槟硞€(gè)結(jié)點(diǎn)變得比它的兩個(gè)子結(jié)點(diǎn)或是其中之一更小了而被打破了,那么我們可以通過將它和它的兩個(gè)子結(jié)點(diǎn)中的較大者交換來恢復(fù)堆。交換可能會在子結(jié)點(diǎn)處繼續(xù)打破堆的有序狀態(tài),因此我們都需要用相同的方式來將其修復(fù),將結(jié)點(diǎn)向下移動直到它的子結(jié)點(diǎn)都比它更小或者是達(dá)到了堆的底部。由位置為k的結(jié)點(diǎn)的子結(jié)點(diǎn)為2k和2k+1可以直接得到對應(yīng)的代碼。方法名為了形象表示這個(gè)過程稱為sink()下沉,即當(dāng)一個(gè)結(jié)點(diǎn)優(yōu)先級太低時(shí)它需要沉(sink)到堆的更低層,

碼實(shí)現(xiàn)如下:

private void sink(int k) {
while(2 * k <= N) {
int j = 2 * k;
if(j < N && less(j, j + 1)) j++;
if(!less(k, j)) break;
exch(k, j);
k = j;
}
}

插入元素

我們將新元素插入到數(shù)組末尾,增加堆的大小并讓這個(gè)元素上浮到合適的位置(如下圖左半部分所示)

刪除最大元素

我們從數(shù)組頂端刪去最大的元素并將數(shù)組的最后一個(gè)元素放到頂端,減小堆的大小并讓這個(gè)元素下沉到合適的位置(如下圖右半部分所示)

基于堆的優(yōu)先隊(duì)列

優(yōu)先隊(duì)列由一個(gè)基于堆的完全二叉樹表示,存儲于數(shù)組pq[1..N]中,pq[0]沒有使用,pq[0]的值有時(shí)可以作為哨兵來用。在insert()中,我們將N加1并把新元素添加在數(shù)組最后,然后用swim()恢復(fù)堆的秩序。在delMax()中,我們從pq[1]中得到需要返回的元素,然后將pq[N]移動到pq[1],將N減1并用sink()來恢復(fù)堆的秩序。同時(shí)我們還將不再使用的pq[N+1]設(shè)為null,以便系統(tǒng)回收它所占用的空間。此處省略動態(tài)調(diào)整數(shù)組大小的代碼,有興趣的朋友可以自己寫寫。

命題:對于一個(gè)含有N個(gè)元素的基于堆的優(yōu)先隊(duì)列,插入元素操作只需不超過(lgN+1)次比較,刪除最大元素的操作需要不超過2lgN次比較

基于堆的優(yōu)先隊(duì)列詳細(xì)代碼如下:

public class MaxPQ<Key extends Comparable<Key>> {
private Key[] pq; //基于堆的完全二叉樹
private int N = 0; //存儲于pa[1..N]中,pq[0]沒有使用
public MaxPQ(int maxN) {
pq = (Key[]) new Comparable[maxN + 1];
this.N = maxN;
}
public boolean isEmpty() {
return N == 0;
}
public int size() {
return N;
}
public void insert(Key v) {
pq[++N] = v;
swim(N);
}
public Key delMax(Key v) {
Key max = pq[1]; //根節(jié)點(diǎn)得到最大元素
exch(1, N--); //將其和最后一個(gè)結(jié)點(diǎn)交換
pq[N + 1] = null; //防止對象游離
sink(1); //恢復(fù)堆的有序性
return max;
}
private boolean less(int i, int j) {
return pq[i].compareTo(pq[j]) < 0;
}
private void exch(int i, int j) {
Key t = pq[i];
pq[i] = pq[j];
pq[j] = t;
}
private void swim(int k) {
while(k > 1 && less(k / 2, k)) {
exch(k / 2, k);
k = k / 2;
}
}
private void sink(int k) {
while(2 * k <= N) {
int j = 2 * k;
if(j < N && less(j, j + 1)) j++;
if(!less(k, j)) break;
exch(k, j);
k = j;
}
}
}

過程:

堆排序

堆排序可以分為兩個(gè)階段。在堆的構(gòu)造階段中,我們將原始數(shù)組重新組織安排進(jìn)一個(gè)堆中;然后在下沉排序階段,我們從堆中按遞減順序取出所有元素并得到排序結(jié)果。為了和我們學(xué)過的代碼保持一致,我們將使用一個(gè)面向最大元素的優(yōu)先隊(duì)列并重復(fù)刪除最大元素。我們不再使將優(yōu)先隊(duì)列的具體表示隱藏,并將直接使用swim()和sink()操作。這樣我們在排序時(shí)就可以將需要排序的數(shù)組本身作為堆,因此無需任何額外空間。

注意:將對應(yīng)下標(biāo)減1可以得到與其他排序算法一樣的排序結(jié)果。否則將是忽略下標(biāo)0的排序算法

到此這篇關(guān)于C++高級數(shù)據(jù)結(jié)構(gòu)之優(yōu)先隊(duì)列的文章就介紹到這了,更多相關(guān) C++優(yōu)先隊(duì)列內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • C++內(nèi)存分區(qū)模型超詳細(xì)講解

    C++內(nèi)存分區(qū)模型超詳細(xì)講解

    在了解內(nèi)存分區(qū)之前,我們先來聊一聊為什么要進(jìn)行內(nèi)存分區(qū)。在進(jìn)行了內(nèi)存分區(qū)之后,在不同的區(qū)域存放的數(shù)據(jù),會有不同的生命周期,從而會讓程序員的編程變得更加靈活
    2022-11-11
  • C語言超市管理系統(tǒng)設(shè)計(jì)

    C語言超市管理系統(tǒng)設(shè)計(jì)

    這篇文章主要為大家詳細(xì)介紹了C語言超市管理系統(tǒng)設(shè)計(jì),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2018-02-02
  • C語言實(shí)現(xiàn)通訊錄的詳細(xì)代碼

    C語言實(shí)現(xiàn)通訊錄的詳細(xì)代碼

    本文詳細(xì)講解了C語言實(shí)現(xiàn)通訊錄的方法,文中通過示例代碼介紹的非常詳細(xì)。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2021-12-12
  • C語言+MySQL實(shí)現(xiàn)推箱子游戲

    C語言+MySQL實(shí)現(xiàn)推箱子游戲

    經(jīng)典的推箱子是一個(gè)來自日本的古老游戲,目的是在訓(xùn)練你的邏輯思考能力。本文將通過C語言和MySQL實(shí)現(xiàn)推箱子這一經(jīng)典游戲,感興趣的可以了解一下
    2022-02-02
  • 淺談C++如何求等差素?cái)?shù)列

    淺談C++如何求等差素?cái)?shù)列

    這篇文章主要介紹了淺談C++如何求等差素?cái)?shù)列,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-07-07
  • 詳解C語言-二級指針三種內(nèi)存模型

    詳解C語言-二級指針三種內(nèi)存模型

    這篇文章主要介紹了詳解C語言-二級指針三種內(nèi)存模型的相關(guān)知識,文中代碼非常詳細(xì),供大家參考和學(xué)習(xí),感興趣的朋友可以了解下
    2020-06-06
  • C語言實(shí)現(xiàn)流星雨效果流程

    C語言實(shí)現(xiàn)流星雨效果流程

    C本篇文章帶你用C語言去實(shí)現(xiàn)漫天流星雨的效果,代碼寫的很清晰,效果非常棒,另有視頻詳解整個(gè)過程,相信你一定能看懂,感興趣的童鞋快來看看吧
    2021-11-11
  • 淺談內(nèi)聯(lián)函數(shù)與宏定義的區(qū)別詳解

    淺談內(nèi)聯(lián)函數(shù)與宏定義的區(qū)別詳解

    本篇文章是對內(nèi)聯(lián)函數(shù)與宏定義的區(qū)別進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下
    2013-05-05
  • C++如何動態(tài)的生成對象詳解

    C++如何動態(tài)的生成對象詳解

    C++是不支持根據(jù)類名動態(tài)地生成對象的,比如從一個(gè)文本文件中讀取類名然后構(gòu)造一個(gè)對象.主要原因是沒有豐富的動態(tài)元信息,沒有單根類庫。那么下面這篇文章就來給大家介紹C++是如何動態(tài)的生成對象,有需要的朋友們可以參考借鑒。
    2017-02-02
  • OpenCV實(shí)現(xiàn)圖像切割功能

    OpenCV實(shí)現(xiàn)圖像切割功能

    這篇文章主要為大家詳細(xì)介紹了OpenCV實(shí)現(xiàn)圖像切割功能,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2019-01-01

最新評論