成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專(zhuān)欄INFORMATION COLUMN

數(shù)據(jù)結(jié)構(gòu)——樹(shù)

Backache / 1565人閱讀

摘要:首先,我們了解一下關(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

相關(guān)文章

  • 【EASYDOM系列教程】之DOM 樹(shù)結(jié)構(gòu)

    摘要:簡(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)。 例如下面這段代...

    nemo 評(píng)論0 收藏0
  • 學(xué)習(xí)javascript數(shù)據(jù)結(jié)構(gòu)(四)——樹(shù)

    摘要:原文博客地址學(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)指正。 ...

    Dean 評(píng)論0 收藏0
  • FineReport中如何制作樹(shù)數(shù)據(jù)集來(lái)實(shí)現(xiàn)組織樹(shù)報(bào)表

    摘要:?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...

    vvpale 評(píng)論0 收藏0
  • FineReport中樹(shù)數(shù)據(jù)集如何實(shí)現(xiàn)組織樹(shù)報(bào)表

    摘要:組織樹(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ù)...

    Travis 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<