[C++]AVL樹怎么轉

AVL樹是啥

一提到AVL樹,腦子里不是旋了,就是懸了。

AVL樹之所以難,并不是因為結構難以理解,而是因為他的旋轉。

AVL樹定義

  • 平衡因子:對于一顆二叉樹,某節點的左右子樹高度之差,就是該節點的平衡因子
  • AVL樹:對于一顆二叉樹,任意節點的平衡因子bf 在范圍[-1,1]之間(即左右子樹高度差的絕對值<=1),則該樹就是平衡二叉樹

為何會有AVL樹

AVL樹是二叉搜索樹的衍生,其名字來源是根據兩位俄羅斯的數學家G.M.Adelson-VelskiiE.M.Landis,他們在1962年發明的一種用來解決二叉搜索樹在極端情況下時間復雜度變為O(n)的情況。而其解決該情況的方法便是:通過旋轉旋轉來調整二叉搜索樹的平衡

AVL樹的節點

AVL樹的節點實現方式有很多,這里采取下面的方法定義節點:

template<class T>struct AVLTreeNode{AVLTreeNode(const T& data): _left(nullptr),_right(nullptr),_parent(nullptr),_data(data),_bf(0){}AVLTreeNode<T>* _left;  // 該節點的左孩子AVLTreeNode<T>* _right; // 該節點的右孩子AVLTreeNode<T>* _parent;// 該節點的父親T _data;//存儲數據int _bf;//平衡因子  
};

這里AVL樹節點的實現增加了_bf (平衡因子)成員變量,用來記錄每個點的平衡因子。同時用了父節點的指針_parent ,目的是方便調整平衡因子。

AVL樹的插入

這里AVL樹的插入操作與二叉搜索樹一樣,唯一不同的是,在插入后要調整平衡因子

  • 對于插入的節點,因為其是插入到葉節點位置,所以他的平衡因子為0
  • 將節點插入后,樹的高度可能會發生改變,此時則要調整他父親,甚至還要調整父親的父親的平衡因子。主要分為以下幾種情況

情況1:

在這里插入圖片描述

如果在節點的右孩子插入,則該節點的bf需要+1(節點70、50的bf連鎖著發生了變化,后面有說明)


情況2:

在這里插入圖片描述

如果在節點的左孩子插入,則該節點的bf需要-1(節點70、50的bf連鎖著發生了變化,后面有說明)

但是沒完!

如果節點的平衡因子因為插入而變成了1 或者-1 ,則說明子樹的高度發生了變化,此時該節點的父節點bf也應發生變化(如果父節點的bf更改之后影響了“爺爺節點”,則爺爺節點也要跟著跟著變化)。所以上圖中的70和50兩節點的bf也要發生變化。


但是還沒完!

bf 的值有可能會變為2、-2,則此時就需要進行旋轉操作(旋轉完成后,會手動調整bf,保證其符合要求)

代碼:

//插入操作
...
//調整平衡因子
cur = newnode;
parent = newnode->_parent;
while (parent)
{if (parent->_right == cur){++parent->_bf;}else if (parent->_left == cur){--parent->_bf;}if (parent->_bf == 0)//AVL樹穩定了,break出去break;else if (parent->_bf == 1 || parent->_bf == -1)//無需旋轉但是需要更改父親的平衡因子{cur = parent;parent = parent->_parent;}else if ()//需要旋轉{}
}

AVL樹的旋轉

重中之重,也是難中之難

AVL樹旋轉的目的

AVL樹是一個平衡二叉搜索樹,因為某些插入操作導致它不再平衡(具體表現是平衡因子變為2或-2),則此時為了使之平衡,就需要進行旋轉操作

AVL樹旋轉操作

AVL樹的旋轉主要分為4種情況:

  1. 左旋
  2. 右旋
  3. 左旋再右旋(雙旋)
  4. 右旋再左旋(雙旋)

情況1:左旋

在這里插入圖片描述

對于插入后父親的bf=2,右孩子的bf=1的情況,采用左旋。且左旋后平衡因子都是0

在這里插入圖片描述


情況2:右旋

在這里插入圖片描述

對于插入后父親的bf=-2,左孩子的bf=-1的情況,采用右旋。且右旋后平衡因子都是0

在這里插入圖片描述


雙旋

這種情況的發生是因為發現對該節點無論左旋還是右旋,都無法使其平衡,原因是插入節點的位置比較特殊

情況3:左旋再右旋

在這里插入圖片描述

對于插入(這里插入b還是c只會影響旋轉完后的平衡因子)后父親的bf=-2,左孩子的bf=1的情況,采用左旋再右旋。這里的平衡因子更新規則與前不同,詳見后文。

在這里插入圖片描述


情況4:先右旋再左旋

在這里插入圖片描述

對于插入(這里插入b還是c只會影響旋轉完后的平衡因子)后父親的bf=2,右孩子的bf=-1的情況,采用右旋再左旋。這里的平衡因子更新規則與前不同,詳見后文。
在這里插入圖片描述

雙旋的平衡因子更新規則

更新規則相比旋轉規則就簡單許多

分為以下3種情況(以上圖為例):

  1. 如果60的平衡因子(旋轉前)是-1,則更新后的60的平衡因子變為0,左孩子變為0,右孩子變為1
  2. 如果60的平衡因子(旋轉前)是1,則更新后60的平衡因子變為0,左孩子變為-1,右孩子變為0
  3. 如果60的平衡因子(旋轉前)是0,則更新后60的平衡因子變為0,左孩子變為0,右孩子變為0

代碼:

這里只給出一部分代碼,剩下的可以自己嘗試寫出來,然后可以到倉庫對照

bool insert(const pair<K,V>& kv)
{//插入Node* newnode = new Node(kv);if (newnode == nullptr) return false;if (_root == nullptr){_root = newnode;return true;}Node* cur = _root;Node* parent = nullptr;while (cur){parent = cur;if (kv.first > cur->_kv.first){cur = cur->_right;}else cur = cur->_left;}//調整節點連接if (kv.first > parent->_kv.first)parent->_right = newnode;else parent->_left = newnode;newnode->_parent = parent;//調整平衡因子cur = newnode;parent = newnode->_parent;while (parent){if (parent->_right == cur){++parent->_bf;}else if (parent->_left == cur){--parent->_bf;}if (parent->_bf == 0)//AVL樹穩定了,break出去break;else if (parent->_bf == 1 || parent->_bf == -1)//無需旋轉但是需要更改父親的平衡因子{cur = parent;parent = parent->_parent;}else if (parent->_bf == 2 || parent->_bf == -2)//需要旋轉{if (parent->_bf == 2 && cur->_bf == 1){RotateL(parent);}else if (parent->_bf == -2 && cur->_bf == -1){RotateR(parent);}else if (parent->_bf == 2 && cur->_bf == -1){RotateRL(parent);}else if (parent->_bf == -2 && cur->_bf == 1){RotateLR(parent);}else{assert(false);}}}return true;
}
//左旋
void RotateL(Node* parent)
{Node* subR = parent->_right;Node* subRL = subR->_left;Node* pparent = parent->_parent;parent->_parent = subR;subR->_left = parent;parent->_right = subRL;if (subRL){subRL->_parent = parent;}if (!pparent){_root = subR;subR->_parent = nullptr;}else{if (pparent->_left == parent){pparent->_left = subR;}else{pparent->_right = subR;}subR->_parent = pparent;}parent->_bf = 0;subR->_bf = 0;
}
//先左旋,再右旋
void RotateLR(Node* parent)
{Node* subL = parent->_left;Node* subLR = subL->_right;int bf = subLR->_bf;RotateL(parent->_left);RotateR(parent);if (bf == 0){subLR->_bf = 0;parent->_bf = 0;subL->_bf = 0;}else if (bf == 1){subLR->_bf = 0;parent->_bf = 0;subL->_bf = -1;}else if (bf == -1){subLR->_bf = 0;parent->_bf = 1;subL->_bf = 0;}else{assert(false);}
}

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

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

相關文章

5、云原生安全之falco的規則解讀(部分)(上)

文章目錄 1、自定義規則測試1.1、自定義檢測定時任務的規則2、自帶規則詳解部分2.1、意外的出站連接源(類似的還有入站連接)2.2、檢測目錄穿越攻擊2.3、rpm數據庫被修改2.4、數據庫派生新的進程2.5、特權容器啟動2.6、啟動容器掛載到敏感路徑2.7、匹配所有在pod內啟動、并連接…

音視頻數字化(數字與模擬-照相機)

目錄 1、模擬/數字 2、第一臺照相機 3、照相機原理 4、取景方式 5、底片 6、數碼相機 7、數碼相機指標 8、數碼相機分類 (1)單反相機 (2)單電相機 (3)無反相機

2024.03.02藍橋云課筆記

1.scanf與printf取消分隔符的限制方法 示例代碼&#xff1a; int main() { char s[10];scanf("%d[^\n]",s);printf("%s",s);return 0; } 運行&#xff1a; 輸入&#xff1a;Hello World 輸出&#xff1a;Hello World 注&#xff1a;其中[]中是一個正則…

(UE4升級UE5)Selected Level Actor節點升級到UE5

本問所用工具為&#xff1a;AssetDeveTool虛幻開發常用工具https://gf.bilibili.com/item/detail/1104960041 在UE4中 編輯器藍圖有個節點為 Get Selected Level Actors 但在UE5中&#xff0c;藍圖直接升級后&#xff0c;節點失效&#xff0c;如圖&#xff1a; 因為在UE5中&am…

Vue3中Vuex狀態管理庫學習筆記

1.什么是狀態管理 在開發中&#xff0c;我們會的應用程序需要處理各種各樣的數據&#xff0c;這些數據需要保存在我們應用程序的某個位置&#xff0c;對于這些數據的管理我們就稱之為狀態管理。 在之前我們如何管理自己的狀態呢&#xff1f; 在Vue開發中&#xff0c;我們使用…

大廠面試經驗:如何對加密后的數據進行模糊查詢操作

加密后的數據對模糊查詢不是很友好&#xff0c;本篇就針對加密數據模糊查詢這個問題來展開講一講實現的思路。 為了數據安全我們在開發過程中經常會對重要的數據進行加密存儲&#xff0c;常見的有&#xff1a;密碼、手機號、電話號碼、詳細地址、銀行卡號、信用卡驗證碼等信息…

YoloV5改進策略:主干網絡改進|MogaNet——高效的多階門控聚合網絡

文章目錄 摘要論文:《MogaNet——高效的多階門控聚合網絡》1、簡介2、相關工作2.1、視覺Transformers2.2、ViT時代的卷積網絡3、從多階博弈論交互的角度看表示瓶頸4、方法論4.1、MogaNet概述4.2、多階門控聚合4.3、通過通道聚合進行多階特征重新分配4.4、實現細節5、實驗5.1、…

Vue 3 中的 setup 函數是如何工作的?

Vue 3 中的 setup 函數是一個新的組件選項&#xff0c;用于使用組合式 API 定義組件的邏輯。這個函數的引入是為了解決 Vue 2 中隨著組件復雜度的增長&#xff0c;選項式的 API 可能導致代碼難以維護和理解的問題。通過 setup 函數&#xff0c;開發者可以更加靈活地組織和共享代…

Python光速入門 - Flask輕量級框架

FlASK是一個輕量級的WSGI Web應用程序框架&#xff0c;Flask的核心包括Werkzeug工具箱和Jinja2模板引擎&#xff0c;它沒有默認使用的數據庫或窗體驗證工具&#xff0c;這意味著用戶可以根據自己的需求選擇不同的數據庫和驗證工具。Flask的設計理念是保持核心簡單&#xff0c…

布隆過濾器實戰

一、背景 本篇文章以解決實際需求的問題的角度進行切入&#xff0c;探討了如果使用布隆過濾器快速丟棄無效請求&#xff0c;降低了系統的負載以及不必要的流量。 我們都知道布隆過濾器是以占用內存小&#xff0c;同時也能夠實現快速的過濾從而滿足我們的需求&#xff0c;本篇…

Matlab偏微分方程擬合 | 源碼分享 | 視頻教程

專欄導讀 作者簡介&#xff1a;工學博士&#xff0c;高級工程師&#xff0c;專注于工業軟件算法研究本文已收錄于專欄&#xff1a;《復雜函數擬合案例分享》本專欄旨在提供 1.以案例的形式講解各類復雜函數擬合的程序實現方法&#xff0c;并提供所有案例完整源碼&#xff1b;2.…

反編譯代碼格式處理

反編譯代碼格式處理 背景解決方案程序跑之后idea格式化 總結 背景 想看看公司里一個工具的代碼實現&#xff0c;手里只有一個jar包&#xff0c;只能通過jd-gui反編譯代碼。但是呢&#xff0c;源碼是有了&#xff0c;但是看的很難受。 解決方案 /*** 替換 {code searchDir}中…

LeetCode 100231.超過閾值的最少操作數 I

給你一個下標從 0 開始的整數數組 nums 和一個整數 k 。 一次操作中&#xff0c;你可以刪除 nums 中的最小元素。 你需要使數組中的所有元素都大于或等于 k &#xff0c;請你返回需要的 最少 操作次數。 示例 1&#xff1a; 輸入&#xff1a;nums [2,11,10,1,3], k 10 輸…

Linux課程四課---Linux開發環境的使用(自動化構建工具-make/Makefile的相關)

作者前言 &#x1f382; ??????&#x1f367;&#x1f367;&#x1f367;&#x1f367;&#x1f367;&#x1f367;&#x1f367;&#x1f382; ?&#x1f382; 作者介紹&#xff1a; &#x1f382;&#x1f382; &#x1f382; &#x1f389;&#x1f389;&#x1f389…

用Java語言創建的Spring Boot項目中,如何傳遞數組呢??

問題&#xff1a; 用Java語言創建的Spring Boot項目中&#xff0c;如何傳遞數組呢&#xff1f;&#xff1f; 在這個思路中&#xff0c;其實&#xff0c;Java作為一個后端開發的語言&#xff0c;沒必要著重于如何傳入&#xff0c;我們主要做的便是對傳入的數組數據進行處理即可…

解決虛擬機啟動報錯:“End kernel panic - not syncing: attempted to kill the idle task”

原本能正常運行的虛擬機&#xff0c;很長一段時間沒用后&#xff0c;今天再次啟動&#xff0c;然后就出現下面的問題&#xff1a; 然后走了一些彎路&#xff0c;比如說刪除該虛擬機然后新建一個虛擬機&#xff08;問題未解決&#xff09;、直接刪除VitualBox重新安裝&#xff0…

感染了后綴為.faust勒索病毒如何應對?數據能夠恢復嗎?

導言&#xff1a; 在網絡安全領域&#xff0c;.faust勒索病毒是近期備受關注的一種惡意軟件。這種病毒采用高度復雜的加密算法&#xff0c;將受感染計算機上的文件全部加密&#xff0c;并要求受害者支付贖金以獲取解密密鑰。.faust勒索病毒的攻擊方式通常是通過電子郵件附件、…

快遞平臺獨立版小程序源碼|帶cps推廣營銷流量主+前端

源碼介紹&#xff1a; 快遞代發快遞代寄寄件小程序可以對接易達云洋一級總代 快遞小程序&#xff0c;接入云洋/易達物流接口&#xff0c;支持選擇快遞公司&#xff0c;三通一達&#xff0c;極兔&#xff0c;德邦等&#xff0c;功能成熟 如何收益: 1.對接第三方平臺成本大約4元…

Python基本數據類型介紹

Python 解釋 Python是一種高級編程語言&#xff0c;以其簡潔、易讀和易用而聞名。它是一種通用的、解釋型的編程語言&#xff0c;適用于廣泛的應用領域&#xff0c;包括軟件開發、數據分析、人工智能等。python是一種解釋型&#xff0c;面向對象、動態數據類型的高級程序設計…

00X集——vba獲取CAD圖中圖元類名objectname

在CAD中&#xff0c;通過快捷鍵PL&#xff08;即POLYLINE命令&#xff09;繪制的線屬于AcDbPolyline。AcDbPolyline也被稱為LWPOLYLINE&#xff0c;即簡單Polyline&#xff0c;它所包含的對象在本身內部。 此外&#xff0c;CAD中還有另一種二維多段線對象&#xff0c;稱為AcDb2…