|
|
# 小結
|
|
|
|
|
|
### 重點回顧
|
|
|
|
|
|
- 二元樹是一種非線性資料結構,體現“一分為二”的分治邏輯。每個二元樹節點包含一個值以及兩個指標,分別指向其左子節點和右子節點。
|
|
|
- 對於二元樹中的某個節點,其左(右)子節點及其以下形成的樹被稱為該節點的左(右)子樹。
|
|
|
- 二元樹的相關術語包括根節點、葉節點、層、度、邊、高度和深度等。
|
|
|
- 二元樹的初始化、節點插入和節點刪除操作與鏈結串列操作方法類似。
|
|
|
- 常見的二元樹型別有完美二元樹、完全二元樹、完滿二元樹和平衡二元樹。完美二元樹是最理想的狀態,而鏈結串列是退化後的最差狀態。
|
|
|
- 二元樹可以用陣列表示,方法是將節點值和空位按層序走訪順序排列,並根據父節點與子節點之間的索引對映關係來實現指標。
|
|
|
- 二元樹的層序走訪是一種廣度優先搜尋方法,它體現了“一圈一圈向外擴展”的逐層走訪方式,通常透過佇列來實現。
|
|
|
- 前序、中序、後序走訪皆屬於深度優先搜尋,它們體現了“先走到盡頭,再回溯繼續”的走訪方式,通常使用遞迴來實現。
|
|
|
- 二元搜尋樹是一種高效的元素查詢資料結構,其查詢、插入和刪除操作的時間複雜度均為 $O(\log n)$ 。當二元搜尋樹退化為鏈結串列時,各項時間複雜度會劣化至 $O(n)$ 。
|
|
|
- AVL 樹,也稱平衡二元搜尋樹,它透過旋轉操作確保在不斷插入和刪除節點後樹仍然保持平衡。
|
|
|
- AVL 樹的旋轉操作包括右旋、左旋、先右旋再左旋、先左旋再右旋。在插入或刪除節點後,AVL 樹會從底向頂執行旋轉操作,使樹重新恢復平衡。
|
|
|
|
|
|
### Q & A
|
|
|
|
|
|
**Q**:對於只有一個節點的二元樹,樹的高度和根節點的深度都是 $0$ 嗎?
|
|
|
|
|
|
是的,因為高度和深度通常定義為“經過的邊的數量”。
|
|
|
|
|
|
**Q**:二元樹中的插入與刪除一般由一套操作配合完成,這裡的“一套操作”指什麼呢?可以理解為資源的子節點的資源釋放嗎?
|
|
|
|
|
|
拿二元搜尋樹來舉例,刪除節點操作要分三種情況處理,其中每種情況都需要進行多個步驟的節點操作。
|
|
|
|
|
|
**Q**:為什麼 DFS 走訪二元樹有前、中、後三種順序,分別有什麼用呢?
|
|
|
|
|
|
與順序和逆序走訪陣列類似,前序、中序、後序走訪是三種二元樹走訪方法,我們可以使用它們得到一個特定順序的走訪結果。例如在二元搜尋樹中,由於節點大小滿足 `左子節點值 < 根節點值 < 右子節點值` ,因此我們只要按照“左 $\rightarrow$ 根 $\rightarrow$ 右”的優先順序走訪樹,就可以獲得有序的節點序列。
|
|
|
|
|
|
**Q**:右旋操作是處理失衡節點 `node`、`child`、`grand_child` 之間的關係,那 `node` 的父節點和 `node` 原來的連線不需要維護嗎?右旋操作後豈不是斷掉了?
|
|
|
|
|
|
我們需要從遞迴的視角來看這個問題。右旋操作 `right_rotate(root)` 傳入的是子樹的根節點,最終 `return child` 返回旋轉之後的子樹的根節點。子樹的根節點和其父節點的連線是在該函式返回後完成的,不屬於右旋操作的維護範圍。
|
|
|
|
|
|
**Q**:在 C++ 中,函式被劃分到 `private` 和 `public` 中,這方面有什麼考量嗎?為什麼要將 `height()` 函式和 `updateHeight()` 函式分別放在 `public` 和 `private` 中呢?
|
|
|
|
|
|
主要看方法的使用範圍,如果方法只在類別內部使用,那麼就設計為 `private` 。例如,使用者單獨呼叫 `updateHeight()` 是沒有意義的,它只是插入、刪除操作中的一步。而 `height()` 是訪問節點高度,類似於 `vector.size()` ,因此設定成 `public` 以便使用。
|
|
|
|
|
|
**Q**:如何從一組輸入資料構建一棵二元搜尋樹?根節點的選擇是不是很重要?
|
|
|
|
|
|
是的,構建樹的方法已在二元搜尋樹程式碼中的 `build_tree()` 方法中給出。至於根節點的選擇,我們通常會將輸入資料排序,然後將中點元素作為根節點,再遞迴地構建左右子樹。這樣做可以最大程度保證樹的平衡性。
|
|
|
|
|
|
**Q**:在 Java 中,字串對比是否一定要用 `equals()` 方法?
|
|
|
|
|
|
在 Java 中,對於基本資料型別,`==` 用於對比兩個變數的值是否相等。對於引用型別,兩種符號的工作原理是不同的。
|
|
|
|
|
|
- `==` :用來比較兩個變數是否指向同一個物件,即它們在記憶體中的位置是否相同。
|
|
|
- `equals()`:用來對比兩個物件的值是否相等。
|
|
|
|
|
|
因此,如果要對比值,我們應該使用 `equals()` 。然而,透過 `String a = "hi"; String b = "hi";` 初始化的字串都儲存在字串常數池中,它們指向同一個物件,因此也可以用 `a == b` 來比較兩個字串的內容。
|
|
|
|
|
|
**Q**:廣度優先走訪到最底層之前,佇列中的節點數量是 $2^h$ 嗎?
|
|
|
|
|
|
是的,例如高度 $h = 2$ 的滿二元樹,其節點總數 $n = 7$ ,則底層節點數量 $4 = 2^h = (n + 1) / 2$ 。
|