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

資訊專欄INFORMATION COLUMN

數(shù)據(jù)結(jié)構(gòu)與算法對的javaScript描述-二叉搜索樹

forrest23 / 1417人閱讀

摘要:定義樹以及相關(guān)概念二叉搜索樹的定義首先是二叉樹最多有兩個節(jié)點(diǎn),分別是左右節(jié)點(diǎn)子節(jié)點(diǎn)的位置是確定的,變現(xiàn)為左子節(jié)點(diǎn)的值小于其父節(jié)點(diǎn)右子節(jié)點(diǎn)的值大于其父節(jié)點(diǎn)在中的描述描述的完整代碼傳送門可視化傳送門節(jié)點(diǎn)類樹是由節(jié)點(diǎn)組成的,要實現(xiàn)樹那么先要實現(xiàn)節(jié)

定義 樹以及相關(guān)概念

二叉搜索樹的定義

首先是二叉樹

最多有兩個節(jié)點(diǎn),分別是左右節(jié)點(diǎn)

子節(jié)點(diǎn)的位置是確定的,變現(xiàn)為

左子節(jié)點(diǎn)的值小于其父節(jié)點(diǎn)

右子節(jié)點(diǎn)的值大于其父節(jié)點(diǎn)

BST在JS中的描述
JS描述的完整代碼傳送門
可視化BST傳送門
節(jié)點(diǎn)類 Node
樹是由節(jié)點(diǎn)組成的,要實現(xiàn)樹那么先要實現(xiàn)節(jié)點(diǎn)
節(jié)點(diǎn)的要素

data:每個節(jié)點(diǎn)都需要有一個數(shù)值

left:左子節(jié)點(diǎn),在沒有左子節(jié)點(diǎn)的時候指向空對象null

right: 右子節(jié)點(diǎn),在沒有右子節(jié)點(diǎn)的時候指向空對象null

count: 計數(shù)值,累計插入的次數(shù),這個是非必須的,作為一個要素是為了后續(xù)處理插入重復(fù)值的問題

show: 顯示方法,展示該節(jié)點(diǎn)的值(和插入次數(shù)),在這里返回一個包含值(和插入次數(shù))的對象

js的描述
class Node {
  constructor (data,
               count = 1,
               left = null,
               right = null)
  {
    // 數(shù)值
    this.data = data
    // 出現(xiàn)次數(shù)
    this.count = count
    // 左右節(jié)點(diǎn)指向
    this.left = left
    this.right = right
  }
  show () {
    return {
      data: this.data,
      count: this.count
    }
  }
}
BST 二叉搜索樹類 屬性

目前我們只需要一個根節(jié)點(diǎn)的屬性,因此一個基本的BST可以描述為:

class BST {
  constructor () {
    // 初始化跟節(jié)點(diǎn)為null
    this.root = null
  }
}
插入方法

我們有了一個基本的BST,這時候我們可以new一個 bst,那么我們怎么插入數(shù)據(jù)呢?這時候我們需要一個insert方法,這個方法有以下的條件:

插入的是節(jié)點(diǎn),也就是上述的類Node的一個實例

如果沒有根節(jié)點(diǎn),bst的根節(jié)點(diǎn)指向該節(jié)點(diǎn)

如果有根節(jié)點(diǎn)則向下遍歷,找到合適的位置插入該節(jié)點(diǎn),遍歷規(guī)則如下圖:

帶有插入方法的BSTjs的描述如下
class BST {
  constructor () {
    // 初始化跟節(jié)點(diǎn)為null
    this.root = null
  }
  /**
   * 插入數(shù)據(jù)
   * @param data
   */
  insert (data) {
    let n = new Node(data, 1)
    if (this.root === null) {
      // 沒有根節(jié)點(diǎn),新的樹把待插入的值作為根節(jié)點(diǎn)
      this.root = n
    } else {
      // 有根節(jié)點(diǎn),遍歷樹直到找到合適的位置
      let current = this.root
      while (true) {
        if (data < current.data) {
          if (current.left === null) {
            current.left = n
            break
          }
          current = current.left
        } else if (data === current.data) {
          current.count += 1
          break
        } else {
          if (current.right === null) {
            current.right = n
            break
          }
          current = current.right
        }
      }
    }
  }
}
插入一組測試數(shù)據(jù)測試
let testData = [
  43,
  34,
  67,
  23,
  34,
  45,
  2,
  78,
  34
]

let bst = new BST()
console.log(JSON.stringify(bst))

for (let data of testData) {
  bst.insert(data)
}
console.log(JSON.stringify(bst))

插入數(shù)據(jù)前:

{"root":null}

插入數(shù)據(jù)后

{
    "root": {
        "data": 43,
        "count": 1,
        "left": {
            "data": 34,
            "count": 3,
            "left": {
                "data": 23,
                "count": 1,
                "left": {
                    "data": 2,
                    "count": 1,
                    "left": null,
                    "right": null
                },
                "right": {
                    "data": 28,
                    "count": 1,
                    "left": null,
                    "right": null
                }
            },
            "right": null
        },
        "right": {
            "data": 67,
            "count": 1,
            "left": {
                "data": 45,
                "count": 1,
                "left": null,
                "right": null
            },
            "right": {
                "data": 78,
                "count": 1,
                "left": null,
                "right": null
            }
        }
    }
}

插入數(shù)據(jù)之后我們是通過nodejs的logger來查看bst,事實上,我們還需要其他的遍歷方法來查看bst

BST的遍歷
遍歷二叉樹通常有三種遍歷方法,分別是中序、先序和后序,他們的遍歷路徑不一樣
中序遍歷
中序遍歷應(yīng)該是最常用的一種遍歷方法

js中的描述

/**
   * 中序遍歷
   * @param node
   */
  inOrder (node) {
    if (node !== null) {
      this.inOrder(node.left)
      console.log(`data:${node.data},count:${node.count}`)
      this.inOrder(node.right)
    }
  }

上述例子中的輸出結(jié)果

中序
data:2,count:1
data:23,count:1
data:28,count:1
data:34,count:3
data:43,count:1
data:45,count:1
data:67,count:1
data:78,count:1

路徑圖

先序遍歷

js中的描述

  /**
   * 先序遍歷
   * @param node
   */
  preOrder (node) {
    if (node !== null) {
      console.log(`data:${node.data},count:${node.count}`)
      this.preOrder(node.left)
      this.preOrder(node.right)
    }
  }

上述例子中的輸出結(jié)果

先序
data:43,count:1
data:34,count:3
data:23,count:1
data:2,count:1
data:28,count:1
data:67,count:1
data:45,count:1
data:78,count:1

路徑圖

后序遍歷

js中的描述

  /**
   * 后序遍歷
   * @param node
   */
  postOrder (node) {
    if (node !== null) {
      this.postOrder(node.left)
      this.postOrder(node.right)
      console.log(`data:${node.data},count:${node.count}`)
    }
  }

上述例子中的輸出結(jié)果

后序
data:2,count:1
data:28,count:1
data:23,count:1
data:34,count:3
data:45,count:1
data:78,count:1
data:67,count:1
data:43,count:1

路徑圖

查找
在二叉搜索樹中查找數(shù)據(jù)非常簡單
最大值與最小值
最小值為樹中的最左邊的葉子節(jié)點(diǎn)得到值,最大值為最右邊子節(jié)點(diǎn)的值
  /**
   * 查找最小值
   * @returns {CanvasPixelArray|string|Object[]|*}
   */
  getMin (node) {
    let current = node || this.root
    while (current.left !== null) {
      current = current.left
    }
    return current
  }
  
  /**
   * 查找最大值
   * @returns {CanvasPixelArray|string|Object[]|*}
   */
  getMax (node) {
    let current = node || this.root
    while (current.right !== null) {
      current = current.right
    }
    return current
  }
查找指定數(shù)據(jù)
根據(jù)數(shù)據(jù)的大小判斷向左向右查找使得查找非常有效率
  /**
   * 查找數(shù)據(jù)
   * @param data
   * @returns {*}
   */
  find (data) {
    let current = this.root,
      result = null
    while (current !== null) {
      if (data === current.data) {
        result = current
        break
      } else if (data < current.data) {
        current = current.left
      } else {
        current = current.right
      }
    }
    return result
  }
bst.getMax().data // 78
bst.getMax().count // 1
bst.getMin().data // 2
bst.getMin().count // 1
刪除數(shù)據(jù)
刪除數(shù)據(jù)其實是操作二叉搜索樹中最麻煩的一部分

我的思路如下:

待刪除節(jié)點(diǎn)沒有子節(jié)點(diǎn):

父節(jié)點(diǎn)鏈向該節(jié)點(diǎn)的鏈接指向null

待刪除節(jié)點(diǎn)只有左節(jié)點(diǎn)或者右節(jié)點(diǎn)

父節(jié)點(diǎn)鏈向他的鏈接指向他的子節(jié)點(diǎn)

待刪除節(jié)點(diǎn)既有左節(jié)點(diǎn)又有右節(jié)點(diǎn)

用他的右子樹的最小節(jié)點(diǎn)取代(值和計數(shù)替代)待刪除節(jié)點(diǎn)

在他的右子樹中刪除右子樹中的最小節(jié)點(diǎn)

基于此,JS中的描述為

  /**
   * 刪除數(shù)據(jù)
   * @param data
   */
  remove ( data ) {
    this.root = this.removeDataFromNode(this.root, data)
  }
  
  /**
   * 從指定節(jié)點(diǎn)中刪除數(shù)據(jù)
   * @param node
   * @param data
   * @returns {*}
   */
  removeDataFromNode (node, data) {
    if (node !== null) {
      if (data === node.data) {
        if (node.left === null && node.right === null) {
        //  沒有子節(jié)點(diǎn)
          return null
        }
        if (node.right === null) {
        //  只有左節(jié)點(diǎn)
          return node.left
        }
        if (node.left === null) {
        //  只有右節(jié)點(diǎn)
          return node.right
        }
        // 有做節(jié)點(diǎn)和右節(jié)點(diǎn)
        // 取右節(jié)點(diǎn)的最小值
        let tempNode = this.getMin(node.right)
        node.data = tempNode.data
        node.count = tempNode.count
        node.right = this.removeDataFromNode(node.right, tempNode.data)
        return node
      } else if (data < node.data) {
        node.left = this.removeDataFromNode(node.left, data)
        return node
      } else {
        node.right = this.removeDataFromNode(node.right, data)
        return node
      }
    } else {
      return null
    }
  }

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/90427.html

相關(guān)文章

  • Python數(shù)據(jù)結(jié)構(gòu)——二叉搜索的實現(xiàn)(上)

    摘要:圖展示了二叉搜索樹的這一特性,顯示的鍵沒有關(guān)聯(lián)任何的值。因為我們必須能夠創(chuàng)建和使用一個空的二叉搜索樹,所以我們將使用兩個類來實現(xiàn),第一個類我們稱之為,第二個類我們稱之為。圖說明了將新節(jié)點(diǎn)插入到一個二叉搜索樹的過程。 二叉搜索樹 我們已經(jīng)知道了在一個集合中獲取鍵值對的兩種不同的方法?;貞浺幌逻@些集合是如何實現(xiàn)ADT(抽象數(shù)據(jù)類型)MAP的。我們討論兩種ADT MAP的實現(xiàn)方式,基于列表的...

    AbnerMing 評論0 收藏0
  • 數(shù)據(jù)結(jié)構(gòu)算法:二分查找

    摘要:為檢查長度為的列表,二分查找需要執(zhí)行次操作。最后需要指出的一點(diǎn)是高水平的讀者可研究一下二叉樹關(guān)于二叉樹,戳這里數(shù)據(jù)結(jié)構(gòu)與算法二叉樹算法常見練習(xí)在一個二維數(shù)組中,每一行都按照從左到右遞增的順序排序,每一列都按照從上到下遞增的順序排序。 常見數(shù)據(jù)結(jié)構(gòu) 簡單數(shù)據(jù)結(jié)構(gòu)(必須理解和掌握) 有序數(shù)據(jù)結(jié)構(gòu):棧、隊列、鏈表。有序數(shù)據(jù)結(jié)構(gòu)省空間(儲存空間?。?無序數(shù)據(jù)結(jié)構(gòu):集合、字典、散列表,無序...

    zsirfs 評論0 收藏0
  • 數(shù)據(jù)結(jié)構(gòu)算法:二分查找

    摘要:為檢查長度為的列表,二分查找需要執(zhí)行次操作。最后需要指出的一點(diǎn)是高水平的讀者可研究一下二叉樹關(guān)于二叉樹,戳這里數(shù)據(jù)結(jié)構(gòu)與算法二叉樹算法常見練習(xí)在一個二維數(shù)組中,每一行都按照從左到右遞增的順序排序,每一列都按照從上到下遞增的順序排序。 常見數(shù)據(jù)結(jié)構(gòu) 簡單數(shù)據(jù)結(jié)構(gòu)(必須理解和掌握) 有序數(shù)據(jù)結(jié)構(gòu):棧、隊列、鏈表。有序數(shù)據(jù)結(jié)構(gòu)省空間(儲存空間?。?無序數(shù)據(jù)結(jié)構(gòu):集合、字典、散列表,無序...

    you_De 評論0 收藏0
  • 數(shù)據(jù)結(jié)構(gòu)算法:二分查找

    摘要:為檢查長度為的列表,二分查找需要執(zhí)行次操作。最后需要指出的一點(diǎn)是高水平的讀者可研究一下二叉樹關(guān)于二叉樹,戳這里數(shù)據(jù)結(jié)構(gòu)與算法二叉樹算法常見練習(xí)在一個二維數(shù)組中,每一行都按照從左到右遞增的順序排序,每一列都按照從上到下遞增的順序排序。 常見數(shù)據(jù)結(jié)構(gòu) 簡單數(shù)據(jù)結(jié)構(gòu)(必須理解和掌握) 有序數(shù)據(jù)結(jié)構(gòu):棧、隊列、鏈表。有序數(shù)據(jù)結(jié)構(gòu)省空間(儲存空間?。?無序數(shù)據(jù)結(jié)構(gòu):集合、字典、散列表,無序...

    gotham 評論0 收藏0
  • 數(shù)據(jù)結(jié)構(gòu)算法JavaScript (不定時更新)

    摘要:每個列表中的數(shù)據(jù)項稱為元素。棧被稱為一種后入先出,的數(shù)據(jù)結(jié)構(gòu)。散列使用的數(shù)據(jù)結(jié)構(gòu)叫做散列表。不包含任何成員的集合稱為空集,全集則是包含一切可能成員的集合。因此二叉搜索樹需要平衡,即左右子樹高度要相近。 樓樓非計算機(jī)專業(yè),但是對計算機(jī)也還算喜歡。個人理解若有偏差,歡迎各位批評指正! 對于數(shù)據(jù)結(jié)構(gòu)和算法一直是我的薄弱環(huán)節(jié),相信大多數(shù)前端工程師可能多少會有些這方面的弱點(diǎn),加上數(shù)據(jù)結(jié)構(gòu)和算法本...

    levius 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<