摘要:首先,我們了解一下關(guān)于樹(shù)的基本知識(shí)每一個(gè)樹(shù)都包含一系列的父子關(guān)系的節(jié)點(diǎn),每個(gè)節(jié)點(diǎn)都有一個(gè)父節(jié)點(diǎn)和若干的子節(jié)點(diǎn)零個(gè)或者多個(gè)沒(méi)有父節(jié)點(diǎn)的節(jié)點(diǎn)稱(chēng)作是根節(jié)點(diǎn)一個(gè)節(jié)點(diǎn)可以有祖先和后代,子樹(shù)由節(jié)點(diǎn)和他的后代構(gòu)成,節(jié)點(diǎn)的一個(gè)屬性是深度深度就是一個(gè)節(jié)點(diǎn)祖先
首先,我們了解一下關(guān)于樹(shù)的基本知識(shí):
每一個(gè)樹(shù)都包含一系列的父子關(guān)系的節(jié)點(diǎn),每個(gè)節(jié)點(diǎn)都有一個(gè)父節(jié)點(diǎn)和若干的子節(jié)點(diǎn)(零個(gè) 或者 多個(gè))
沒(méi)有父節(jié)點(diǎn)的節(jié)點(diǎn)稱(chēng)作是根節(jié)點(diǎn)
一個(gè)節(jié)點(diǎn) 可以有祖先 和 后代,子樹(shù)由節(jié)點(diǎn)和他的后代構(gòu)成,節(jié)點(diǎn)的一個(gè)屬性是深度:深度就是一個(gè)節(jié)點(diǎn)祖先節(jié)點(diǎn)的數(shù)量
、樹(shù)的高度是 所有節(jié)點(diǎn)深度中的最大值
二叉樹(shù)和二叉樹(shù)的搜索二叉樹(shù):就是每個(gè)節(jié)點(diǎn)最多只能有兩個(gè)節(jié)點(diǎn) 一個(gè)是左側(cè)的節(jié)點(diǎn) 另一個(gè)是右側(cè)的節(jié)點(diǎn)
二叉樹(shù)搜索又是一種特殊的二叉樹(shù):只允許在左側(cè)存儲(chǔ)比父節(jié)點(diǎn)小的值 在右側(cè)存儲(chǔ)比父節(jié)點(diǎn)大的值,如下面的圖所示:
由上圖可以看出每一個(gè)節(jié)點(diǎn)都有兩個(gè)指針,一個(gè)是指向左側(cè)節(jié)點(diǎn),一個(gè)是指向右側(cè)節(jié)點(diǎn)
由此,我們可以發(fā)現(xiàn)有關(guān)樹(shù)的規(guī)則:
樹(shù)的左側(cè)所有的節(jié)點(diǎn)的值肯定是小于根節(jié)點(diǎn)的值,樹(shù)的右側(cè)肯定是大于根節(jié)點(diǎn)的值
所以當(dāng)刪除一個(gè)擁有兩個(gè)節(jié)點(diǎn)的節(jié)點(diǎn)時(shí),為了遵循樹(shù)的規(guī)律,需要找到節(jié)點(diǎn)右側(cè)樹(shù)的最小節(jié)點(diǎn)值
接下來(lái),我們將通過(guò)javascript 創(chuàng)建一個(gè)二叉樹(shù)類(lèi),二叉樹(shù)是由一個(gè)個(gè)節(jié)點(diǎn)組成的,所以,在類(lèi)中需要有一個(gè)節(jié)點(diǎn)類(lèi),根節(jié)點(diǎn)屬性,還有基本的操作方法,分別是 插入,刪除,遍歷(前序,中序,后序),查詢(xún)(最大值,最小值,和某一指定值)
//創(chuàng)建一個(gè)二叉樹(shù)類(lèi) function BinarySearchTree(key){ //節(jié)點(diǎn)類(lèi),有值,左節(jié)點(diǎn),右節(jié)點(diǎn)三個(gè)公有屬性 var Node = function(key){ this.key = key; this.left = null; this.rigth = null; } //在樹(shù)中插入一個(gè)節(jié)點(diǎn) this.insert = function(newNode){ if(root === null){ root = newNode; }else{ insertNode(root,newNode); } } //在樹(shù)中查找一個(gè)鍵,如果鍵存在就返回true this.search = function(key){ return searchNode(root,key); } //通過(guò)中序遍歷所有的節(jié)點(diǎn) this.inOrderTraverse = function(){ inOrderTraverseNode(root,callback); } //通過(guò)先序遍歷所有的節(jié)點(diǎn) this.preOrderTravers = function(){ preOrderTraverseNode(root,callback); } //通過(guò)后續(xù)遍歷所有的節(jié)點(diǎn) this.postOrderTravers = function(){ postOrderTraverseNode(root,callback); } //返回樹(shù)中最小的鍵值 this.min = function(){ return minNode(root); } //返回樹(shù)中最大的鍵值 this.max = function(){ return maxNode(root); } //從樹(shù)中移除某一個(gè)鍵值 this.remove = function(key){ root = removeNode(root,key); } var root = null; }操作二叉樹(shù)的方法及其實(shí)現(xiàn)
首先,實(shí)現(xiàn) insert方法,大致思路是這樣的:先判斷 根節(jié)點(diǎn)是否為空,如果為空的話(huà),就直接將新的節(jié)點(diǎn)作為根節(jié)點(diǎn),如果不為空的話(huà)就調(diào)用一個(gè)插入節(jié)點(diǎn)的私有方法。
//創(chuàng)建插入節(jié)點(diǎn)的方法" this.insert = function(newNode){ if(root === null){ root = newNode; }else{ insertNode(root,newNode); } }
insertNode的具體實(shí)現(xiàn)是這樣的:
var insertNode = function(node,newNode){ if(newNode.key < node.key){ if(node.left === null){ node.left = newNode; }else{ insertNode(node.left,newNode); } }else{ if(node.right === null){ node.right = newNode; }else{ insertNode(node.right,newNode); } } }
函數(shù)解釋?zhuān)夯舅枷刖褪?將節(jié)點(diǎn) 他的左節(jié)點(diǎn) 和 他的右節(jié)點(diǎn) 看作是一個(gè)小的子樹(shù),先判斷新節(jié)點(diǎn)的鍵值 是否小于節(jié)點(diǎn)的鍵值,如果小,就判斷左節(jié)點(diǎn)是否為空,如果左節(jié)點(diǎn)為空,就將新的節(jié)點(diǎn)直接賦值給左節(jié)點(diǎn),如果左節(jié)點(diǎn)不為空,就將左節(jié)點(diǎn)作為搜索比較的基點(diǎn),調(diào)用自身再進(jìn)行搜索比較(遞歸算法),相反如果大于節(jié)點(diǎn)的鍵值,再判斷右節(jié)點(diǎn)是否為空,如果為空就直接將節(jié)點(diǎn)賦值給右節(jié)點(diǎn),如果節(jié)點(diǎn)不為空同樣調(diào)用自身。
接下來(lái),我們定義 中序遍歷的函數(shù):
中序遍歷是二叉樹(shù)的一種遍歷方式之一,實(shí)質(zhì)上就是以從小到大的順序訪(fǎng)問(wèn)二叉樹(shù),中序遍歷的一種應(yīng)用就是對(duì)樹(shù)中存儲(chǔ)的值進(jìn)行從小到大的排列,中序遍歷的輔助方法如下所述:
var inOrderTraverseNode = function(node,callback){ if(node !== null){ inOrderTraverseNode(node.left,callback);(1) callback(node.key);(2) inOrderTraverseNode(node.right.callback);(3) } }
我以一個(gè)例子畫(huà)了調(diào)用過(guò)程(簡(jiǎn)單粗糙,后期會(huì)修改):
先序遍歷的作用是打印樹(shù)的一個(gè)結(jié)構(gòu)化文檔,他的輔助函數(shù)如下面所述:
var preOrderTraverseNode = function(node,callback){ if(node !== null){ callback(node.key); preOrderTraverseNode(node.left,callback); preOrderTraverseNode(node.right,callback); } }
調(diào)用過(guò)程圖 與 中序遍歷相似
后序遍歷的作用的計(jì)算樹(shù)的子目錄 和 父目錄的所有文件所占空間的大小,其輔助函數(shù)的具體實(shí)現(xiàn)如下面所示:
var postOrderTraverseNode = function(node,callback){ if(node!== null){ postOrderTraverseNode(node.left,callback); postOrderTraverseNode(node.right.callback); callback(node.key); } }
以上我們對(duì)三種遍歷方式做了具體的實(shí)現(xiàn),接下來(lái)我們完成對(duì)樹(shù)中的所有值進(jìn)行搜索。
基于構(gòu)建樹(shù)時(shí)所遵循的具體規(guī)則,我們可以看出在樹(shù)的最左端是樹(shù)的最小值,樹(shù)的最右端就是樹(shù)的最大值
那么尋找最小值的輔助函數(shù)是這樣實(shí)現(xiàn)的:
//搜索最小值時(shí)的搜索函數(shù) var minNode = function(node){ //判斷節(jié)點(diǎn)是否存在,如果存在就執(zhí)行循環(huán) if(node){ while(node && node.left !== null){ node = node.left; } return node.key; } return null; }
函數(shù)解釋:首先判斷所傳的節(jié)點(diǎn)(一般是根節(jié)點(diǎn),看是否存在),如果存在的話(huà)就進(jìn)行循環(huán),直到找到最左端的節(jié)點(diǎn)(節(jié)點(diǎn)的左節(jié)點(diǎn)不存在,那么這就是最左邊的節(jié)點(diǎn)),返回節(jié)點(diǎn)的所有值
尋找最大值的輔助函數(shù)是這樣實(shí)現(xiàn)的:
//搜索樹(shù)中的最大值函數(shù) var maxNode = function(node){ if(node){ while(node && node.right !== null){ node = node.right; } return node.key; } return null; }
與尋找最小值相反,該函數(shù)是循環(huán)找最右邊的節(jié)點(diǎn)
搜索特定值得輔助函數(shù)如下:
//搜索特定值得輔助函數(shù) var searchNode = fucntion(node,key){ var result; if(node === null){ result = false; } if(key < node.key){ searchNode(node.left,key); }else if(key > node.key){ searchNode(node.right,key); }else{ result = true; } return result; }
函數(shù)解釋?zhuān)喝绻胍獙ふ业膋ey值小于節(jié)點(diǎn)值就在節(jié)點(diǎn)的右側(cè)樹(shù)上尋找,在節(jié)點(diǎn)不為空的情況下回調(diào)該函數(shù),縮小右側(cè)樹(shù)的尋找范圍,最后節(jié)點(diǎn)如果為空就是沒(méi)有找到,反之,找到;大于節(jié)點(diǎn)值同理
最后,我們完成移除節(jié)點(diǎn)的方法
根據(jù)子節(jié)點(diǎn)的數(shù)量,我們將節(jié)點(diǎn)分為三種情況:
葉子節(jié)點(diǎn),既沒(méi)有左節(jié)點(diǎn)也沒(méi)有右節(jié)點(diǎn)
只有一個(gè)節(jié)點(diǎn),左節(jié)點(diǎn) 或者 右節(jié)點(diǎn)
有兩個(gè)節(jié)點(diǎn)左節(jié)點(diǎn)和右節(jié)點(diǎn)
針對(duì)上面三種情況,有三種刪除節(jié)點(diǎn)的方法
對(duì)于第一種情況:因?yàn)槿~子節(jié)點(diǎn)即沒(méi)有左節(jié)點(diǎn),也沒(méi)有右節(jié)點(diǎn),所以直接將該節(jié)點(diǎn)賦值為null就可以了,但是,單單這樣是遠(yuǎn)遠(yuǎn)不夠的,我們還需要對(duì)其相應(yīng)的指針進(jìn)行處理,對(duì)于葉子節(jié)點(diǎn)來(lái)說(shuō),我們需要處理葉子節(jié)點(diǎn)父節(jié)點(diǎn)的指針,使其的指針指向null,所以需要在執(zhí)行完刪除以后返回當(dāng)前所基于的node節(jié)點(diǎn),層層返回后實(shí)質(zhì)上是以node節(jié)點(diǎn)為root的子樹(shù)
對(duì)于第二種情況:移除只有一個(gè)節(jié)點(diǎn)時(shí)的節(jié)點(diǎn),那么節(jié)點(diǎn)的左側(cè)節(jié)點(diǎn)或者右側(cè)節(jié)點(diǎn)就可以取代節(jié)點(diǎn)的位置
對(duì)于第三種情況:當(dāng)移除有兩個(gè)子節(jié)點(diǎn)的節(jié)點(diǎn)時(shí),需要找到該節(jié)點(diǎn)右側(cè)樹(shù)上節(jié)點(diǎn)值最小的節(jié)點(diǎn),并用該節(jié)點(diǎn)代替它,最后使其的右側(cè)節(jié)點(diǎn)指向刪除節(jié)點(diǎn)后的樹(shù)
//移除一個(gè)節(jié)點(diǎn)的輔助函數(shù) var removeNode = function(node,key){ if(node === null){ return null; } //尋找節(jié)點(diǎn)是key值的節(jié)點(diǎn) if(key < node.key){ node.left = removeNode(node.left,key); return node; }else if(key > node.key){ node.right = removeNode(node.right,key); return node; }else{ //第一種情況——兩個(gè)葉子節(jié)點(diǎn)都沒(méi)有 if(node.left === null && node.right === null){ node = null; return node; } //第二種情況——只有一個(gè)葉子節(jié)點(diǎn) if(node.left === null){ node = node.right; return node; }else if(node.right === null){ node = node.left; return node; } //第三種情況——有兩個(gè)葉子節(jié)點(diǎn)的節(jié)點(diǎn) var min = minNode(node.right); node.key = min; node.right = removeNode(node.right,min); return node; } }
最后,我們?cè)敿?xì)解釋一下第三種情況為什么會(huì)這樣寫(xiě),當(dāng)該節(jié)點(diǎn)被刪除的時(shí)候,我們需要尋找一個(gè)節(jié)點(diǎn)來(lái)代替,尋找哪個(gè)節(jié)點(diǎn)呢?肯定是尋找比他值大的節(jié)點(diǎn),因?yàn)橐WC左節(jié)點(diǎn)的值要小于尋找到的新值,所以我們鎖定的目標(biāo)是節(jié)點(diǎn)的右側(cè)樹(shù),然而找的節(jié)點(diǎn)值還必須小于右節(jié)點(diǎn)的值,所以尋找右側(cè)樹(shù)的最小值,并將該節(jié)點(diǎn)刪除
以上是我對(duì)樹(shù)的理解和總結(jié),本人接受各位同仁大神的指點(diǎn),謝謝
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/81416.html
摘要:簡(jiǎn)單來(lái)說(shuō),節(jié)點(diǎn)作為樹(shù)結(jié)構(gòu)中的連接點(diǎn),最終構(gòu)成了完整的樹(shù)結(jié)構(gòu)。節(jié)點(diǎn)樹(shù)結(jié)構(gòu)通過(guò)節(jié)點(diǎn)概念,我們可以將原本的樹(shù)結(jié)構(gòu)改成節(jié)點(diǎn)樹(shù)結(jié)構(gòu)進(jìn)行表示。節(jié)點(diǎn)之間的關(guān)系中的表示模型,也可以用來(lái)表示節(jié)點(diǎn)樹(shù)結(jié)構(gòu)中節(jié)點(diǎn)之間的關(guān)系。值得注意的是和元素并不是兄弟關(guān)系。 DOM 樹(shù)結(jié)構(gòu) DOM 之所以可以訪(fǎng)問(wèn)和更新 HTML 頁(yè)面中的內(nèi)容、結(jié)構(gòu)和樣式,是因?yàn)?DOM 將 HTML 頁(yè)面解析為一個(gè) 樹(shù)結(jié)構(gòu)。 例如下面這段代...
摘要:原文博客地址學(xué)習(xí)數(shù)據(jù)結(jié)構(gòu)四樹(shù)知乎專(zhuān)欄簡(jiǎn)書(shū)專(zhuān)題前端進(jìn)擊者知乎前端進(jìn)擊者簡(jiǎn)書(shū)博主博客地址的個(gè)人博客人之所能,不能兼?zhèn)?,棄其所短,取其所長(zhǎng)。通常子樹(shù)被稱(chēng)作左子樹(shù)和右子樹(shù)。敬請(qǐng)期待數(shù)據(jù)結(jié)構(gòu)篇最后一篇文章學(xué)習(xí)數(shù)據(jù)結(jié)構(gòu)五圖參考文章樹(shù)數(shù)據(jù)結(jié)構(gòu)二叉樹(shù) 前言 總括: 本文講解了數(shù)據(jù)結(jié)構(gòu)中的[樹(shù)]的概念,盡可能通俗易懂的解釋樹(shù)這種數(shù)據(jù)結(jié)構(gòu)的概念,使用javascript實(shí)現(xiàn)了樹(shù),如有紕漏,歡迎批評(píng)指正。 ...
摘要:?jiǎn)栴}描述,組織樹(shù)報(bào)表中由與父來(lái)實(shí)現(xiàn)組織樹(shù)報(bào)表,若層級(jí)數(shù)較多時(shí),對(duì)每個(gè)單元格設(shè)置過(guò)濾條件和形態(tài)會(huì)比較繁瑣,因此提供了一種特殊的數(shù)據(jù)集樹(shù)數(shù)據(jù)集,只需要簡(jiǎn)單的設(shè)置就能自動(dòng)遞歸出層級(jí),方便的實(shí)現(xiàn)如下圖組織樹(shù)報(bào)表圖一圖二構(gòu)建樹(shù)新建報(bào)表,添加數(shù)據(jù)集新建 問(wèn)題描述FineReport,組織樹(shù)報(bào)表中由id與父id來(lái)實(shí)現(xiàn)組織樹(shù)報(bào)表,若層級(jí)數(shù)較多時(shí),對(duì)每個(gè)單元格設(shè)置過(guò)濾條件和形態(tài)會(huì)比較繁瑣,因此FineR...
摘要:組織樹(shù)報(bào)表中由與父來(lái)實(shí)現(xiàn)組織樹(shù)報(bào)表,若層級(jí)數(shù)較多時(shí),對(duì)每個(gè)單元格設(shè)置過(guò)濾條件和形態(tài)會(huì)比較繁瑣,因此提供了一種特殊的數(shù)據(jù)集樹(shù)數(shù)據(jù)集,只需要簡(jiǎn)單的設(shè)置就能自動(dòng)遞歸出層級(jí),方便的實(shí)現(xiàn)如下圖組織樹(shù)報(bào)表圖一圖二構(gòu)建樹(shù)新建報(bào)表,添加數(shù)據(jù)集新建工作薄,添 組織樹(shù)報(bào)表中由id與父id來(lái)實(shí)現(xiàn)組織樹(shù)報(bào)表,若層級(jí)數(shù)較多時(shí),對(duì)每個(gè)單元格設(shè)置過(guò)濾條件和形態(tài)會(huì)比較繁瑣,因此FineReport提供了一種特殊的數(shù)據(jù)...
閱讀 1461·2021-11-25 09:43
閱讀 2601·2021-09-24 10:30
閱讀 3671·2021-09-06 15:02
閱讀 3610·2019-08-30 15:55
閱讀 3310·2019-08-30 15:53
閱讀 1705·2019-08-30 15:52
閱讀 2151·2019-08-30 14:21
閱讀 2019·2019-08-30 13:55