首頁 後端開發 C++ C++中的二元堆和二元搜尋樹

C++中的二元堆和二元搜尋樹

Aug 22, 2023 pm 04:10 PM
c++ 二元搜尋樹 二元堆

C++中的二元堆和二元搜尋樹

在C 程式設計中,二元堆和二元搜尋樹是兩個常用的資料結構,它們具有相似之處,但也有著不同點。本文將分別介紹二元堆和二元搜尋樹的概念、基本操作及其應用場景。

一、二元堆

1.1 概念

二元堆是一種完全二元樹,滿足以下兩種性質:

1.1.1 堆序性

堆序性指在一個二元堆中,每個節點的值都不大於(或不小於)其父節點的值。這裡以最大堆為例,根節點的值是整個樹中最大的值,而所有子節點的值都小於等於根節點的值。

1.1.2 完全二元樹性質

除了最底層之外,其他層都必須填滿,且所有節點都必須向左對齊。

這裡以如下的陣列來表示一個最大堆:

[ 16, 14, 10, 8, 7, 9, 3, 2, 4 , 1 ]

則其對應的堆疊如下圖所示:

16

/
14 10
/ /
8 7 9 3
/
2 4

1

1.2 基本操作

1.2.1 插入操作

在一個二元堆中插入一個新元素的操作,採用「上濾」(sift up)的方法進行調整:

  • 將新元素插入到堆底最左邊的空位;
  • #將新元素和它的父節點進行比較,如果新元素的值比其父節點大,則交換兩者的位置,重複此過程直到新元素不再比其父節點大,或到達堆頂。

1.2.2 刪除操作

在一個二元堆中刪除堆頂元素的操作,並採用「下濾」(sift down)的方法進行調整:

  • 將堆頂元素和堆底最右邊的元素互換;
  • 刪除原來的堆頂元素;
  • 將新的堆頂元素逐層與其子節點進行比較,如果它的值小於子節點中的最大值,則將它與子節點中的最大值交換,重複此過程直到滿足堆序性。

1.3 應用場景

二元堆疊常用來實作優先佇列,以及基於堆的排序演算法,如堆排序、topK問題等。

二、二元搜尋樹

2.1 概念

二元搜尋樹(Binary Search Tree,BST)是一種有序樹,滿足以下性質:

2.1.1 左子樹上所有節點的值都小於它的根節點的值。

2.1.2 右子樹上所有節點的值都大於它的根節點的值。

2.1.3 左、右子樹也分別為二元搜尋樹。

如下樹為例:

    6
  /   
 2     7

/
1 4 9

   /    /
  3   5 8

則它是一棵二元搜尋樹。

2.2 基本操作

2.2.1 尋找操作

在在二元搜尋樹中尋找一個節點的操作,其實質就是不斷地比較要查找的節點值與當前節點值的大小,並沿著左/右子樹遞歸查找。

2.2.2 插入操作

在二元搜尋樹中插入新節點的操作,需要從根節點開始進行比較,並找到其應該插入的位置,插入後需要滿足二元搜尋樹的性質。

2.2.3 刪除操作

在二元搜尋樹中刪除一個節點的操作,分成三種情況:

  • 要刪除的節點是葉子節點,直接刪除即可;
  • 要刪除的節點只有一個子節點時,用它的子節點替換該節點;
  • 要刪除的節點有兩個子節點時,用該節點右子樹的最小節點代替該節點,並將該節點右子樹的最小節點刪除。

2.3 應用場景

二元搜尋樹常用於實作字典、符號表等具有尋找和插入操作的場景,其中的尋找效能與資料的分佈有關。

三、二元堆和二元搜尋樹的比較

3.1 相似之處

二元堆和二元搜尋樹都是二元樹,具有一些相同的性質:

  • 根節點的初始位置皆可以是任意節點;
  • 都可以用來實現優先權佇列;
  • 插入和刪除的時間複雜度均為O(logn)。

3.2 不同之處

二元堆和二元搜尋樹之間也有一些明顯的不同點:

3.2.1 資料分佈

在二元堆中,元素沒有任何規律地分佈在各個節點中,只需保證每個節點都滿足堆序性即可;而在二元搜尋樹中,元素的大小有特定的排序規則,即滿足左小右大的性質。

3.2.2 最小/最大值的存取

在二元堆中,可以O(1)地存取到最大/最小值,即在根節點中得到,但是存取其他元素的時間複雜度為O(logn);而在二元搜尋樹中,找出最小/最大值需要遍歷子樹,時間複雜度也為O(logn)。

3.2.3 刪除和插入操作

在二元堆中,每次刪除和插入操作都必須遵循堆序性,即O(logn)的時間複雜度;而在在二元搜尋樹中,尋找一個節點和插入一個新節點的時間複雜度與樹的高度相關,因此最壞情況下可能需要O(n)的時間複雜度。

3.3 選型建議

在選擇二元堆和二元搜尋樹時,需要根據應用場景的具體情況進行選擇。

如果需要快速地取得最小/最大值,並且對元素的大小沒有特殊要求,可以優先選擇二元堆。

如果需要快速地插入/刪除元素,並且需要元素的大小有一定的排序規律,可以考慮選擇二元搜尋樹。

四、 結論

綜上所述,二元堆和二元搜尋樹都是比較重要的資料結構,在不同的場景下有著各自的優缺點。了解二元堆和二元搜尋樹的概念、基本操作及其應用場景對於編寫高效的程式具有重要的意義。

以上是C++中的二元堆和二元搜尋樹的詳細內容。更多資訊請關注PHP中文網其他相關文章!

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

C初始化技術 C初始化技術 Jul 18, 2025 am 04:13 AM

C 中有多種初始化方式,適用於不同場景。 1.基本變量初始化包括賦值初始化(inta=5;)、構造初始化(inta(5);)和列表初始化(inta{5};),其中列表初始化更嚴格且推薦使用;2.類成員初始化可通過構造函數體賦值或成員初始化列表(MyClass(intval):x(val){}),後者更高效並適用於const和引用成員,C 11還支持類內直接初始化;3.數組和容器初始化可使用傳統方式或C 11的std::array和std::vector,支持列表初始化並提升安全性;4.默認初

在C中解釋RAII 在C中解釋RAII Jul 22, 2025 am 03:27 AM

RAII是C 中用於資源管理的重要技術,其核心在於通過對像生命週期自動管理資源。它的核心思想是:資源在構造時獲取,在析構時釋放,從而避免手動釋放導致的洩漏問題。例如,在沒有RAII時,文件操作需手動調用fclose,若中途出錯或提前return就可能忘記關閉文件;而使用RAII後,如FileHandle類封裝文件操作,離開作用域後會自動調用析構函數釋放資源。 1.RAII應用於鎖管理(如std::lock_guard)、2.內存管理(如std::unique_ptr)、3.數據庫和網絡連接管理等

什麼是虛擬幣高頻交易?高頻交易的原理與技術實現要點 什麼是虛擬幣高頻交易?高頻交易的原理與技術實現要點 Jul 23, 2025 pm 11:57 PM

高頻交易是虛擬幣市場中技術含量最高、資本最密集的領域之一。它是一場關於速度、算法和尖端科技的競賽,普通市場參與者難以涉足。了解其運作方式,有助於我們更深刻地認識到當前數字資產市場的複雜性和專業化程度。對於大多數人而言,認識並理解這一現象,比親自嘗試更為重要。

什麼是C中的破壞者? 什麼是C中的破壞者? Jul 19, 2025 am 03:15 AM

C 中的析構函數是一種特殊的成員函數,會在對象離開作用域或被顯式刪除時自動調用。它的主要作用是清理對像在其生命週期內可能獲取的資源,如內存、文件句柄或網絡連接。析構函數在以下情況下自動調用:局部變量離開作用域時、對指針調用delete時、包含對象的外部對象析構時。定義析構函數時需在類名前加~,且無參數和返回值。若未定義,編譯器會生成默認析構函數,但不會處理動態內存釋放。注意事項包括:每個類只能有一個析構函數,不支持重載;建議將繼承類的析構函數設為virtual;派生類析構函數先執行,再自動調用

C位操作員解釋了 C位操作員解釋了 Jul 18, 2025 am 03:52 AM

C 中的位運算符用於直接操作整數的二進制位,適用於系統編程、嵌入式開發、算法優化等領域。 1.常見的位運算符包括按位與(&)、按位或(|)、按位異或(^)、按位取反(~)、左移()。 2.使用場景有狀態標誌管理、掩碼操作、性能優化以及加密/壓縮算法。 3.注意事項包括區分位運算與邏輯運算、避免對有符號數進行不安全的右移、不過度使用影響可讀性,並建議使用宏或常量提高代碼清晰度、注意操作順序、通過測試驗證行為。

在C中使用STD ::可選 在C中使用STD ::可選 Jul 21, 2025 am 01:52 AM

要判斷std::optional是否有值,可使用has_value()方法或直接在if語句中判斷;返回可能為空的結果時推薦使用std::optional,避免空指針和異常;不應濫用,某些場景下布爾返回值或獨立bool變量更合適;初始化方式多樣,但需注意使用reset()清空值,並留意生命週期和構造行為。

如何將字符串轉換為大寫或C中的小寫字母? 如何將字符串轉換為大寫或C中的小寫字母? Jul 19, 2025 am 01:34 AM

InC ,stringscanbeconvertedtouppercaseorlowercasebyprocessingeachcharacterusingstd::toupperorstd::tolowerfrom1.Casteachcharactertounsignedcharbeforeapplyingthefunctiontoavoidundefinedbehavior.2.Modifycharactersinplaceorcopythestringifpreservingtheori

成員初始化列表 成員初始化列表 Jul 19, 2025 am 02:03 AM

在C 中,成員初始化列表用於在構造函數中初始化成員變量,尤其適用於const成員、引用成員、無默認構造函數的類成員及性能優化。其語法以冒號開頭,後接逗號分隔的初始化項。使用成員初始化列表的原因包括:1.const成員變量必須在初始化時賦值;2.引用成員必須初始化;3.無默認構造函數的類類型成員需顯式調用構造函數;4.提升類類型成員的構造效率。此外,初始化順序由成員在類中聲明順序決定,而非初始化列表中的順序,因此需注意避免依賴未初始化成員。常見應用場景包括初始化常量、引用、複雜對象及需傳參構造的

See all articles