搜索二叉树(C++实现)

简介: 搜索二叉树(C++实现)

二叉搜索树简介

二叉搜索树又称二叉排序树,它或者是一棵空树,或者是具有以下性质的二叉树:

  • 若它的左子树不为空,则左子树上所有节点的值都小于根节点的值
  • 若它的右子树不为空,则右子树上所有节点的值都大于根节点的值
  • 它的左右子树也分别为二叉搜索树

1. 二叉搜索树的查找

  • a、从根开始比较,查找,比根大则往右边走查找,比根小则往左边走查找。
  • b、最多查找高度次,走到到空,还没找到,这个值不存在。

2. 二叉搜索树的插入

插入的具体过程如下:

  • a. 树为空,则直接新增节点,赋值给root指针
  • b. 树不空,按二叉搜索树性质查找插入位置,插入新节点

3. 二叉搜索树删除

  • 删除结点的左子树为空或者删除结点的右子树为空

解决方法:将它的不为空的子树给它的父亲结点

  • 删除结点的左子树和右子树都不为空

替换法:找左子树的最右结点或找右子树的最左结点,先交换,再删除这个结点

应用场景

Key的搜索模型:确定一个值在不在?例如门禁系统

Key/ Value 模型:确定Key在不在,并通过Key查找 value

例如字典,通过汉语查找英语,统计某单词出现的次数

代码实现

注意:代码使用了迭代和循环两种方式来实现二叉搜索树

在循环法来对搜索二叉树进行操作时,需要用一个指针来记录它的父亲,而使用递归法来对搜索二叉树进行操作时,不需要用指针来记录,只需要传参的时候传引用即可,就能直接对指向进行修改!

#pragma once
#include<iostream>
using namespace std;
template <class K>
struct BinarySearchTree
{
  BinarySearchTree(const K& x = K())
    :_left(nullptr)
    , _val(x)
    , _right(nullptr)
  {}
  K _val;
  BinarySearchTree<K>* _left;
  BinarySearchTree<K>* _right;
};
template <class K>
class BSTree
{
  typedef BinarySearchTree<K> BST;
public:
  BSTree()
  {}
  // 循环插入
  bool Insert(const K& x)
  {
    if (_root == nullptr)
    {
      _root = new BinarySearchTree<K>(x);
    }
    else
    {
      BST* cur = _root;
      BST* parent = nullptr;
      while (cur)
      {
        if (cur->_val < x)
        {
          parent = cur;
          cur = cur->_right;
        }
        else if (cur->_val > x)
        {
          parent = cur;
          cur = cur->_left;
        }
        else
        {
          return false;//排序二叉树中不能插入相同的数
        }
      }
      cur = new BinarySearchTree<K>(x);
      if (x > parent->_val)
      {
        parent->_right = cur;
      }
      else
      {
        parent->_left = cur;
      }
    }
    return true;
  }
  //循环查找
  bool* find(const K& x)
  {
    if (_root == nullptr)
    {
      return false;
    }
    BST* cur = _root;
    while (cur)
    {
      if (cur->_val < x)
      {
        cur = cur->_right;
      }
      else if (cur->_val > x)
      {
        cur = cur->_left;
      }
      else
      {
        return true;
      }
    }
    return false;
  }
  // 循环删除
  bool erase(const K& x)
  {
    if (_root == nullptr)
    {
      return false;
    }
    else
    {
      BST* cur = _root;
      BST* parent = cur;
      while (cur)
      {
        if (cur->_val < x)
        {
          parent = cur;
          cur = cur->_right;
        }
        else if (cur->_val > x)
        {
          parent = cur;
          cur = cur->_left;
        }
        else
        {
          // 找到即将要删除的结点了
          //左孩子为空或者右孩子为空的情况
          if (cur->_left == nullptr||cur->_right == nullptr) 
          {
            if (cur->_left == nullptr)
            {
              if (cur == _root)
              {
                _root = cur->_right;
              }
              else
              {
                if (parent->_val < x)
                {
                  parent->_right = cur->_right;
                }
                else
                  parent->_left = cur->_right;
              }
            }
            else
            {
              if (cur == _root)
              {
                _root = cur->_left;
              }
              else
              {
                if (parent->_val < x)
                {
                  parent->_right = cur->_left;
                }
                else
                  parent->_left = cur->_left;
              }
            }
            delete cur;
             
            cur = nullptr;
             
            return true;
          }
          // 左右孩子都存在的情况
          // 替换法
          else
          {
            // cur 为要删除的结点,且左右子树都不为空
            BST* del = cur->_left;
            BST* parent = del;
            while (del->_right)
            {
              parent = del;
              del = del->_right;
            }
            swap(cur->_val, del->_val);
            parent->_right = del->_left;
            delete del;
            del = nullptr;
            return true;
          }
        }
      }
      return false;
    }
  }
  // 前序遍历时,恰好为升序
  void Preorder()
  {
    _Preorder(_root);
    cout << endl;
  }
  // 递归插入
  bool insertR(const K& x)
  {
    if (_root == nullptr)
    {
      _root = new BinarySearchTree<K>(x);
      return true;
    }
    else
    {
      return _insertR(_root, x);
    }
  }
  // 递归查找
  bool findR(const K& x)
  {
    return _findR(_root, x);
  }
  //递归删除
  bool eraseR(const K& x)
  {
    return _eraseR(_root, x);
  }
private:
  void _Preorder(BST* root)
  {
    if (root == nullptr)
    {
      return;
    }
    _Preorder(root->_left);
    cout << root->_val << " ";
    _Preorder(root->_right);
  }
  bool _eraseR(BST*& root, const K& x)
  {
    if (root == nullptr)
      return false;
    if (root->_val < x) return _eraseR(root->_right, x);
    else if (root->_val > x) return _eraseR(root->_left, x);
    else
    {
      if (root->_left==__nullptr || root->_right==nullptr)
      {
        if (root->_left == nullptr)
        {
          BST* del = root;
          root = root->_right;
          delete del;
          return true;
        }
        else
        {
          BST* del = root;
          root = root->_left;
          delete del;
          return true;
        }
      }
      else
      {
        BST*& cur = root->_right;
        while (cur->_left)
        {
          cur = cur->_left;
        }
        swap(cur->_val, root->_val);
        return _eraseR(root->_right, x);
      }
    }
  }
  bool _findR(BST*root, const K& x)
  {
    if (root == nullptr) return false;
    if (root->_val < x) return _findR(root->_right, x);
    else if (root->val > x) return _findR(root->_left, x);
    else return true;
  }
  bool _insertR(BST*& root, const K& x)
  {
    if (root == nullptr)
    {
      root = new BinarySearchTree<K>(x);
      return true;
    }
    if (root->_val < x)
    {
      return _insertR(root->_right, x);
    }
    else if (root->_val > x)
    {
      return _insertR(root->_left, x);
    }
    else
      return false;
  }
  BST*_root = nullptr;
};
相关文章
|
8月前
|
C++
二叉树进阶面试题(精华总结)【C++版本】
二叉树进阶面试题(精华总结)【C++版本】
|
8月前
|
存储 编译器 数据库
【C/C++ 数据结构 】线索二叉树全解析:从数学原理到C++实现
【C/C++ 数据结构 】线索二叉树全解析:从数学原理到C++实现
183 0
|
8月前
|
算法 C++ 开发者
【C/C++ 数据结构 】二叉树基本性质:对于任何一颗二叉树T,若其终端结点为n0 ,那么度数为2的结点数为n2。则n0=n2+1...
【C/C++ 数据结构 】二叉树基本性质:对于任何一颗二叉树T,若其终端结点为n0 ,那么度数为2的结点数为n2。则n0=n2+1...
110 0
|
13天前
|
Java C++
【C++数据结构——树】二叉树的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现二叉树的基本运算。​ 相关知识 创建二叉树 销毁二叉树 查找结点 求二叉树的高度 输出二叉树 //二叉树节点结构体定义 structTreeNode{ intval; TreeNode*left; TreeNode*right; TreeNode(intx):val(x),left(NULL),right(NULL){} }; 创建二叉树 //创建二叉树函数(简单示例,手动构建) TreeNode*create
37 12
|
13天前
|
C++
【C++数据结构——树】二叉树的性质(头歌实践教学平台习题)【合集】
本文档介绍了如何根据二叉树的括号表示串创建二叉树,并计算其结点个数、叶子结点个数、某结点的层次和二叉树的宽度。主要内容包括: 1. **定义二叉树节点结构体**:定义了包含节点值、左子节点指针和右子节点指针的结构体。 2. **实现构建二叉树的函数**:通过解析括号表示串,递归地构建二叉树的各个节点及其子树。 3. **使用示例**:展示了如何调用 `buildTree` 函数构建二叉树并进行简单验证。 4. **计算二叉树属性**: - 计算二叉树节点个数。 - 计算二叉树叶子节点个数。 - 计算某节点的层次。 - 计算二叉树的宽度。 最后,提供了测试说明及通关代
37 10
|
13天前
|
存储 算法 测试技术
【C++数据结构——树】二叉树的遍历算法(头歌教学实验平台习题) 【合集】
本任务旨在实现二叉树的遍历,包括先序、中序、后序和层次遍历。首先介绍了二叉树的基本概念与结构定义,并通过C++代码示例展示了如何定义二叉树节点及构建二叉树。接着详细讲解了四种遍历方法的递归实现逻辑,以及层次遍历中队列的应用。最后提供了测试用例和预期输出,确保代码正确性。通过这些内容,帮助读者理解并掌握二叉树遍历的核心思想与实现技巧。
37 2
|
3月前
|
存储 算法 C++
【搜索算法】 跳马问题(C/C++)
【搜索算法】 跳马问题(C/C++)
|
3月前
|
人工智能 算法 Java
【搜索算法】数字游戏(C/C++)
【搜索算法】数字游戏(C/C++)
|
6月前
|
存储 C++
【C++】二叉树进阶之二叉搜索树(下)
【C++】二叉树进阶之二叉搜索树(下)
40 4
|
6月前
|
Java 编译器 C++
【C++】二叉树进阶之二叉搜索树(上)
【C++】二叉树进阶之二叉搜索树(上)
41 3