初識優先級隊列與堆

1.優先級隊列

? ? ? ? 由前文隊列queue可知,隊列是一種先進先出(FIFO)的數據結構,但有些情況下,操作的數據可能帶有優先級,一般出隊列時,可能需要優先級高的元素先出隊列,在此情況下,使用隊列queue顯然不合適,

????????同時在我們生活場景中也有類似于這中優先事情處理的情況,在醫院看病的時候大家都在排隊,突然有一個頭上插了一把刀的老爺爺也來排隊,這時候雖然大家都很急著去看病,但是出于老爺爺情況緊急,所以大家都會默認讓老爺爺優先去見醫生,而不是讓老爺爺按照一般的規則去老老實實的排隊;

? ? ? ? 考慮到隨時會遇到上述這種情況,數據結構應該提供兩個最基本的操作,一個是返回最高優先級對象一個是添加新的對象。這種數據結構就是優先級隊列(Priority Queue)

2.初識堆

????????JDK1.8中的PriorityQueue底層使用了這種數據結構,而堆實際就是在完全二叉樹的基礎上進行了一些調整。

2.1 堆的概念

????????如果有一個關鍵碼的集合K = {k0,k1, k2,…,kn-1},把它的所有元素按完全二叉樹的順序存儲方式存儲 在一 個一維數組中,并滿足:Ki = K2i+1 且 Ki >= K2i+2) i = 0,1,2…,則稱為 小堆(或大 堆)。將根節點最大的堆叫做最大堆或大根堆,根節點最小的堆叫做最小堆或小根堆。

堆的性質

  • 堆中某個節點的值總是不大于或不小于其父節點的值;

  • 堆總是一棵完全二叉樹。

2.1.1 小根堆?

? ? ? ? ? ? ? ? ? ? ? ?

? ? ? ? 如上圖小根堆的邏輯結構所示,在一顆大的完全二叉樹中的小二叉樹中,該小二叉樹的根節點遠遠小于該樹左右子節點,且這時候不考慮左右子節點的數值哪個大;

2.1.2 大根堆

? ? ? ? ? ? ? ??

???????? 如上圖大根堆的邏輯結構所示,給一顆大的完全二叉樹中的小二叉樹中,該小二叉樹的根節點遠遠大于于該樹左右子節點,且這時候不考慮左右子節點的數值哪個大;

2.2 堆的存儲方式

????????從堆的概念可知,堆是一棵完全二叉樹,因此可以層序的規則采用順序的方式來高效存儲,如下圖所示堆的存儲圖解;(堆是將一個一維數組中的數據按照層序遍歷的規則將這些數據存儲到一個完全二叉樹里的完全二叉樹)

????????注意:由該圖中的邏輯結構和存儲結構可知,對于非完全二叉樹,則不適合使用順序方式進行存儲,因為為了通過一維存儲結構能夠還原二叉樹,一維存儲空間中必須要存儲空節點,就會導致該存儲空間利用率比較低。

???????? 將元素存儲到數組中后,可以根據本主初識二叉樹章節的性質5對樹進行還原。

????????假設i為節點在數組中的下標,則有:

????????如果i為0,則i表示的節點為根節點,否則i節點的雙親節點為 (i - 1)/2

????????如果2 * i + 1 小于節點個數,則節點i的左孩子下標為2 * i + 1,否則沒有左孩子

????????如果2 * i + 2 小于節點個數,則節點i的右孩子下標為2 * i + 2,否則沒有右孩子?

2.3 堆的創建(小根堆)

2.3.1 堆向下調整

? ? ? ? 思考:

????????對于集合{ 27,15,19,18,28,34,65,49,25,37 }中的數據,如何將其創建成堆,首先按照堆的規則將以上數劇放入到完全二叉樹里面,如下圖所示:

? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?

????????仔細觀察上圖后發現:當前根節點27的左右子樹已經完全滿足堆的性質,因此只需將根節點向下調整好即可。

? ? ? ? 調整過程如下:

1. 讓parent標記需要調整的節點,child標記parent的左孩子(注意:parent如果有孩子一定先是有左孩子)

2. 如果parent的左孩子存在,即:child < size, 進行以下操作,直到parent的左孩子不存在? ? ? ? ?2.1parent右孩子是否存在,存在找到左右孩子中最小的孩子,讓child進行標

? ? ?2.2將parent與較小的孩子child比較,如果:

? ? ? ? ? ? ? ? 2.2.1 parent小于較小的孩子child,調整結束

? ? ? ? ? ? ? ? 2.2.2 否則:交換parent與較小的孩子child,交換完成之后,parent中大的元素向下移動,可能導致子樹不滿足對的性質,因此需要繼續向下調整,即parent = child;child = parent*2+1(左子樹); 然后繼續2

? ? ? ? 詳細調整圖解如下圖所示:

2.3.2 小根堆代碼實現

package com.bit.demo1;public class MySmallHeap {public void shiftDown(int[] array, int parent) {//child和parent時數組存儲的節點的索引// child先標記parent的左孩子,因為parent可能右左沒有右int child = 2*parent + 1;int size = array.length;//所有節點的數目while(child < size ) {//主要保證當前訪問到的child都在所有節點的范圍內,是有效的//child+1是二叉樹的最后一個節點// 如果右孩子存在,找到左右孩子中較小的孩子,用child進行標記if(child + 1 < size) {if(array[child + 1] < array[child]) {child = child + 1;}}// 如果最小的孩子比其父親還小,說明該結構沒有滿足堆的特性,進行交換if(array[child] < array[parent]) {int tmp = array[parent];array[parent] = array[child];array[child] = tmp;} else {//滿足就退出循環break;}// parent中大的元素往下移動,可能會造成子樹不滿足堆的性質,因此需要繼續向下調整parent = child;child = 2*parent + 1;}}}

2.3.3?小根堆代碼測試

 public static void main(String[] args) {MySmallHeap mySmallHeap = new MySmallHeap();int[] array = {27,15,19,18,28,34,65,49,25,37};System.out.println("調整前:");for(int i = 0; i < array.length ; i++) {System.out.print(array[i] + " ");}for(int parent = (array.length-2)/2 ; parent >= 0; parent --) {mySmallHeap.shiftDown(array, parent);}System.out.println();System.out.println("調整后:");for(int i = 0; i < array.length ; i++) {System.out.print(array[i] + " ");}System.out.println();}

? ? ? ? 關于最初的parent索引 ?

? ? ? ? 測試結果如下圖所示:

?????????注意:在調整以parent為根的二叉樹時,必須要滿足parent的左子樹和右子樹已經是堆了才可以向下調整。

????????最壞的情況即圖示的情況,從根一路比較到葉子,比較的次數為完全二叉樹的高度,即時間復雜度為

2.4 建堆的時間復雜度

2.4.1 建堆的步驟圖解

????????對于普通的序列{ 1,5,3,8,7,6 },我們需要建立大堆,即根節點的左右子樹不滿足堆的特性,又該如何調整呢,其中里面的細節如下:

? ? ? ? 大概思路:

????????找倒數第一個非葉子節點(最右側的最小子樹根節點),從該節點位置開始往前進行按照根堆的規則運作,一直運作到根節點,這時候才確定根節點的數值;

? ? ? ? 因為為了根節點導致下面的不同子樹的結構都發生了變化,所以接下來確定索引為1的根節點的數字,這樣就需要重復上述的步驟;

? ? ? ? 就這樣重復上面兩個步驟,知道確定最右側最小子樹的根節點(索引為(arr.length-2)/2)的數值,我們的整體過程才算結束;

2.4.2 時間復雜度求解圖解

? ? ? ? 至此可得:建堆的時間復雜度為O(N)

ps:本次的內容就到這里了,如果喜歡的話就請一鍵三連哦!!!

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/207676.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/207676.shtml
英文地址,請注明出處:http://en.pswp.cn/news/207676.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

git常用命令指南

目錄 一、基本命令 1、創建分支 2、切換分支 3、合并分支 4、初始化空git倉庫 二、文件操作 1、創建文件 2、添加多個文件 3、查看項目的當前狀態 4、修改文件 5、刪除文件 6、提交項目 三、實際操作 1、創建目錄 2、進入新目錄 3、初始化空git倉庫 4、創建文…

C++STL的string模擬實現

文章目錄 前言string的成員變量成員函數構造函數拷貝構造賦值重載 模擬實現string各種接口print迭代器普通迭代器const迭代器 string比較大小push_backinsert 和 eraseinserterase reserve和resizereserveresize swapfindcout和cincoutcin 前言 今天要講string的底層實現&…

總線(什么是南北橋?您都用過哪些總線?)

什么是總線&#xff1f; 計算機系統中的總線&#xff08;Bus&#xff09;是指計算機設備和設備之間傳輸信息的公共數據通道&#xff0c;是連接計算機硬件系統內多種設備的通信線路&#xff0c;它的一個重要特征是由總線上的所有設備共享&#xff0c;因此可以將計算機系統內的多…

python基于輕量級GhostNet模型開發構建23種常見中草藥圖像識別系統

輕量級識別模型在我們前面的博文中已經有過很多實踐了&#xff0c;感興趣的話可以自行移步閱讀&#xff1a; 《移動端輕量級模型開發誰更勝一籌&#xff0c;efficientnet、mobilenetv2、mobilenetv3、ghostnet、mnasnet、shufflenetv2駕駛危險行為識別模型對比開發測試》 《基…

Vue 核心 數據監聽 computed | watch

Vue 核心 數據監聽 computed | watch 一、今日學習目標 1.指令補充 指令修飾符v-bind對樣式增強的操作v-model應用于其他表單元素 2.computed計算屬性 基礎語法計算屬性vs方法計算屬性的完整寫法成績案例 3.watch偵聽器 基礎寫法完整寫法 4.綜合案例 &#xff08;演示&…

缺陷責任期與質量保修期如何快速區分?

缺陷責任期 《建設工程質量保證金管理辦法》第二條對缺陷給出了定義&#xff0c;是指建設工程質量不符合工程建設強制性標準、設計文件&#xff0c;以及承包合同的約定。缺陷責任期是指承包人對工程質量瑕疵擔保的期限&#xff0c;由發承包雙方在合同中進行約定&#xff0c;期…

制造業數字化轉型該怎么做?這篇1.6萬字的文章終于講透了!

制造業數字化轉型該怎么做&#xff1f;下面通過 1.6W 字干貨內容&#xff0c;全面講透制造業數字化轉型。 &#xff08;為防后續找不到&#xff0c;建議大家先點贊收藏~&#xff09; 引言&#xff1a; 1、發達國家制造業回流力度加大&#xff0c;中國制造業戰略地位提升。 …

selenium 解決 id定位、class定位中,屬性值帶空格的解決辦法

一、前置說明 selenium遇到下面這種元素&#xff1a; <th id"demo id" class"value1 value2 value3 ">1、雖然id一般不會有空格&#xff0c;但是前端錯誤的這種寫法(如下圖)&#xff0c;會造成使用id定位不到元素&#xff0c;如&#xff1a; find…

面試遇到的一些問題(二)

1、v-if v-show 區別,他們的生命周期區別 v-show: (類似于display:none/black 的切換)不管初始值是true 或false 都會進行渲染,狀態改變也不會銷毀和重新生成。不會影響生命周期 v-if : 是根據條件,dom進行刪除插入操作。 依附于普通元素時:會觸發父組件的beforeUpdate和u…

IOday6作業

1>使用有名管道&#xff0c;完成兩個進程的相互通信 //create.c #include<myhead.h>int main(int argc, const char *argv[]) {if((mkfifo("myfifo1",0664)) -1){perror("mkfifo");return -1;}if((mkfifo("myfifo2",0664)) -1){perror…

MYSQL練題筆記-高級查詢和連接-這系列最后一題以及下個系列(子查詢)的第一題

今天做了下面兩題&#xff0c;到第三題的時候想了下但是沒有太多的思路&#xff0c;然后看題解的時候實在是覺得自己不會&#xff0c;打算明天看吧。 1.按分類統計薪水相關的表和題目如下 我是想著簡化問題&#xff0c;先找出薪水低于30000的員工&#xff0c;然后找這些員工的上…

JAVA 鎖

樂觀鎖 樂觀鎖是一種樂觀思想&#xff0c;即認為讀多寫少&#xff0c;遇到并發寫的可能性低&#xff0c;每次去拿數據的時候都認為別人不會修改&#xff0c;所以不會上鎖&#xff0c;但是在更新的時候會判斷一下在此期間別人有沒有去更新這個數據&#xff0c;采取在寫時先讀出…

Sam Altman當選“TIME時代周刊”2023年度最佳CEO!還有梅西、Taylor Swift當選...

TIME時代周刊昨日在官網公布了2023年最佳CEO—— Sam Altman當選! 此外&#xff0c;Taylor Swift當選年度最佳人物&#xff0c;梅西當選年度最佳運動員。 Sam Altman的當選可謂是實至名歸&#xff01;沒有誰能比火爆全球的ChatGPT背后&#xff0c;OpenAI的CEO更“成功”了。 …

ssh安裝及問題解決

ssh安裝及遇到的問題 ssh分為客戶端 openssh-client 和服務器 openssh-server&#xff0c;可以利用以下命令確認是否安裝&#xff1a; dpkg -l | grep ssh我用ubantu安裝的&#xff0c;所以默認安裝了客戶端 安裝客戶端和服務器端的命令分別為&#xff1a; sudo apt-get ins…

金融量化交易:使用Python實現遺傳算法

大家好&#xff0c;遺傳算法是一種受自然選擇過程啟發的進化算法&#xff0c;用于尋找優化和搜索問題的近似解決方案。本文將使用Python來實現一個用于優化簡單交易策略的遺傳算法。 1.遺傳算法簡介 遺傳算法是一類基于自然選擇和遺傳學原理的優化算法&#xff0c;其特別適用…

MySQL 教程 2.1

MySQL 插入數據 MySQL 表中使用 INSERT INTO 語句來插入數據。 你可以通過 mysql> 命令提示窗口中向數據表中插入數據&#xff0c;或者通過PHP腳本來插入數據。 語法 以下為向MySQL數據表插入數據通用的 INSERT INTO SQL語法&#xff1a; INSERT INTO table_name (colu…

使用Pytorch實現Grad-CAM并繪制熱力圖

這篇是我對嗶哩嗶哩up主 霹靂吧啦Wz 的視頻的文字版學習筆記 感謝他對知識的分享 看一下這個main cnn.py的文件 那這里我為了方便 就直接從官方的torch vision這個庫當中導入一些我們常用的model 比如說我這里的例子是采用的mobile net v3 large這個模型 然后這里我將pretrain設…

微信小程序 純css畫儀表盤

剛看到設計稿的時候第一時間想到的就是用canvas來做這個儀表盤&#xff0c;雖然本人的畫布用的不是很好但還可以寫一寫&#x1f600;。話不多說直接上代碼。最后有純css方法 <!--wxml--> <canvas canvas-id"circle" class"circle" >// js dat…

MySQL 忘記root密碼后重置密碼操作

在忘記 MySQL 密碼的情況下&#xff0c;可以通過 --skip-grant-tables 關閉服務器的認證&#xff0c;然后重置 root 的密碼&#xff0c;具體操作步驟如下。 步驟 1)&#xff1a;關閉正在運行的 MySQL 服務。打開 cmd 進入 MySQL 的 bin 目錄。 步驟 2)&#xff1a;輸入mysqld -…

【Docker】容器數據持久化及容器互聯

容器數據持久化及容器互聯 一、Docker容器的數據管理1.1、什么是數據卷1.2、數據卷特點1.3、數據卷使用 二、Docker的數據卷容器2.1、什么是數據卷容器2.2、掛載數據卷容器方法 三、Docker數據卷的備份和還原3.1、數據備份方法3.2、數據還原方法 四、Docker容器互聯4.1、docker…