数据结构——二叉树搜索树(二叉搜索树的概念、实现、先序遍历、中序遍历、后序遍历)

目录

一、二叉搜索树的概念

1、什么是二叉搜索树?

2、二叉搜索树的操作

二、二叉搜索树的实现

1、创建二叉搜索树  向树种插入数据

2、遍历二叉搜索树

1)先序遍历

2)中序遍历

3)后序遍历


一、二叉搜索树的概念

1、什么是二叉搜索树?

二叉搜索树(BST,Binary Search Tree),也称二叉排序树或二叉查找树

 二叉搜索树是一颗二叉树, 可以为空;如果不为空,满足以下性质:

- 非空左子树的所有键值小于其根结点的键值。
- 非空右子树的所有键值大于其根结点的键值。
- 左、右子树本身也都是二叉搜索树。

 二叉搜索树的特点:

- 二叉搜索树的特点就是相对较小的值总是保存在左结点上, 相对较大的值总是保存在右结点上.
- 那么利用这个特点, 我们可以做什么事情呢?
- 查找效率非常高, 这也是二叉搜索树中, 搜索的来源.

2、二叉搜索树的操作

-  `insert(key)`:向树中插入一个新的键。
-  `search(key)`:在树中查找一个键,如果结点存在,则返回`true`;如果不存在,则返回`false`。
-  `preOrderTraverse`:通过先序遍历方式遍历所有结点。
-  `inOrderTraverse`:通过中序遍历方式遍历所有结点。
-  `postOrderTraverse`:通过后序遍历方式遍历所有结点。
-  `min`:返回树中最小的值/键。
-  `max`:返回树中最大的值/键。
-  `remove(key)`:从树中移除某个键。

二、二叉搜索树的实现

1、创建二叉搜索树  向树种插入数据

      class Node {
            constructor(data) {
                this.left = null;
                this.data = data;
                this.right = null;
            }
        }
        class BST {
            constructor() {
                this.root = null;
            }
            insert(ele) {
                // 创建新节点
                let newnode = new Node(ele);
                // console.log(newnode);
                if (this.root == null) { // 空树
                    this.root = newnode
                } else {
                    this.insertNode(this.root, newnode)
                }
            }
            insertNode(root, newnode) {
                if (newnode.data < root.data) { // 放左边
                    if (root.left == null) {
                        root.left = newnode
                    } else {
                        this.insertNode(root.left, newnode)
                    }
                } else { //放右边
                    if (root.right == null) {
                        root.right = newnode
                    } else {
                        this.insertNode(root.right, newnode)
                    }
                }
            }
        }

2、遍历二叉搜索树

1)先序遍历

遍历过程为:

- ①访问根结点;
- ②先序遍历其左子树;
- ③先序遍历其右子树。

2)中序遍历

遍历过程为:

- ①中序遍历其左子树;
- ②访问根结点;
- ③中序遍历其右子树。

3)后序遍历

遍历过程为:

- ①后序遍历其左子树;
- ②后序遍历其右子树;
- ③访问根结点。

 

  class Node {
            constructor(data) {
                this.left = null;
                this.data = data;
                this.right = null;
            }
        }
        class BST {
            constructor() {
                this.root = null;
            }
            insert(ele) {
                // 创建新节点
                let newnode = new Node(ele);
                // console.log(newnode);
                if (this.root == null) { // 空树
                    this.root = newnode
                } else {
                    this.insertNode(this.root, newnode)
                }
            }
            insertNode(root, newnode) {
                if (newnode.data < root.data) { // 放左边
                    if (root.left == null) {
                        root.left = newnode
                    } else {
                        this.insertNode(root.left, newnode)
                    }
                } else { //放右边
                    if (root.right == null) {
                        root.right = newnode
                    } else {
                        this.insertNode(root.right, newnode)
                    }
                }

            }
            // 前序遍历
            preOrderTraversal() {
                this.preOrderTraversalNode(this.root)
            }
            preOrderTraversalNode(root) {
                if (root != null) { // {left:node,data:11,right:node} != null
                    // 1.根
                    console.log(root.data); //11 7  15
                    // 2.前序遍历左子树
                    this.preOrderTraversalNode(root.left)
                    // 3.前序遍历右子树
                    this.preOrderTraversalNode(root.right)
                }
            }
            // 中序遍历
            inOrderTraversal(){
                this.inOrderTraversalNode(this.root)
            }
            inOrderTraversalNode(root){
                if(root !=null){
                    // 1.中序遍历左子树
                    this.inOrderTraversalNode(root.left)
                    // 2.根
                    console.log(root.data);
                    // 3.中序遍历右子树
                    this.inOrderTraversalNode(root.right)
                }
            }
            // 后序遍历
            postOrderTraversal(){
                this.postOrderTraversalNode(this.root)
            }
            postOrderTraversalNode(root){
                if(root !=null){
                    // 1.后序遍历左子树
                    this.postOrderTraversalNode(root.left)
                    
                    // 2.后序遍历右子树
                    this.postOrderTraversalNode(root.right)
                    // 3.根
                    console.log(root.data);
                }
            }
        }
        let bst = new BST();
        bst.insert(11)
        bst.insert(7)
        bst.insert(15)
        bst.insert(5)
        bst.insert(3)
        bst.insert(9)
        bst.insert(8)
        bst.insert(10)
        bst.insert(13)
        bst.insert(12)
        bst.insert(14)
        bst.insert(20)
        bst.insert(18)
        bst.insert(25)
        console.log(bst.postOrderTraversal());
  • 1
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 先序遍历中序遍历后序遍历都是二叉树遍历的方法,它们的区别在于访问根节点的顺序和访问左右子的顺序不同。 1. 先序遍历 先序遍历的访问顺序是:根节点 -> 左子 -> 右子。 例如,对于下面这棵二叉树: ``` 1 / \ 2 3 / \ / \ 4 5 6 7 ``` 其先序遍历序列为:1 2 4 5 3 6 7。 2. 中序遍历 中序遍历的访问顺序是:左子 -> 根节点 -> 右子。 例如,对于下面这棵二叉树: ``` 1 / \ 2 3 / \ / \ 4 5 6 7 ``` 其中序遍历序列为:4 2 5 1 6 3 7。 3. 后序遍历 后序遍历的访问顺序是:左子 -> 右子 -> 根节点。 例如,对于下面这棵二叉树: ``` 1 / \ 2 3 / \ / \ 4 5 6 7 ``` 其后序遍历序列为:4 5 2 6 7 3 1。 希望这个回答能够解决您的疑惑。 ### 回答2: 先序遍历中序遍历后序遍历二叉树的三种常见遍历方式。 先序遍历:先访问根节点,然后遍历左子,最后遍历右子。在遍历的过程中,根节点总是在左子和右子之前被访问到。 中序遍历:先遍历左子,然后访问根节点,最后遍历右子。在遍历的过程中,根节点总是在左子和右子之间被访问到。 后序遍历:先遍历左子,然后遍历右子,最后访问根节点。在遍历的过程中,根节点总是在左子和右子之后被访问到。 以一个简单的二叉树为例,如下所示: A / \ B C / \ \ D E F 先序遍历:A -> B -> D -> E -> C -> F 中序遍历:D -> B -> E -> A -> C -> F 后序遍历:D -> E -> B -> F -> C -> A 其中,先序遍历的访问顺序是根节点->左子->右子中序遍历的访问顺序是左子->根节点->右子后序遍历的访问顺序是左子->右子->根节点。 这三种遍历方式都有自己的应用场景。先序遍历常用于打印表达式,中序遍历常用于二叉搜索树的中序输出,后序遍历常用于计算二叉树的表达式。 总之,先序遍历中序遍历后序遍历是对二叉树的不同遍历顺序,每种遍历方式都有自己的特点和应用场景。 ### 回答3: 先序遍历中序遍历后序遍历都是二叉树遍历的方法。下面分别对这三种遍历进行简单的说明。 先序遍历是指从根节点开始,先遍历根节点,然后按照先序遍历的顺序,递归遍历左子和右子。具体步骤如下:先输出根节点,然后递归遍历左子,最后递归遍历右子中序遍历是指从根节点开始,先递归遍历左子,然后输出根节点,最后再递归遍历右子。具体步骤如下:先递归遍历左子,然后输出根节点,最后递归遍历右子后序遍历是指从根节点开始,先递归遍历左子,然后递归遍历右子,最后输出根节点。具体步骤如下:先递归遍历左子,然后递归遍历右子,最后输出根节点。 这三种遍历方法都是通过递归的方式实现的,其中先序遍历中序遍历的次序不同,而后序遍历的次序更为靠后。不同的遍历方式可以帮助我们了解二叉树结构的不同特点,如先序遍历可以轻松找到二叉树的根节点,中序遍历可以输出二叉树的有序节点序列,后序遍历可以方便地进行二叉树的删除操作。 总结起来,先序遍历中序遍历后序遍历都是二叉树遍历的方法,只不过它们的递归次序不同。熟练掌握这三种遍历方式可以帮助我们更好地理解二叉树的结构和特点。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值