【C++进阶学习】二叉搜索树(2)

简介: 【C++进阶学习】二叉搜索树(2)

4、二叉搜索树的插入


  • 具体操作过程:


  1. 若key大于当前结点的数据域之值,则插入右子树
  2. 若key小于当前结点的数据域之值,则插入左子树
  3. 若key等于当前结点的数据域之值,则插入失败,返回false
  4. 若走到空结点直接插入,插入成功,返回true


  • 示图:插入56



  • 迭代实现:


bool Insert(const K& key)
{
    if (_root == nullptr)
    {
        _root = new Node(key);
        return true;
    }
    Node* cur = _root;
    Node* prev = nullptr;
    while (cur)
    {
        if (cur->_key > key)
        {
            prev = cur;
            cur = cur->_left;
        }
        else if (cur->_key < key)
        {
            prev = cur;
            cur = cur->_right;
        }
        else
        {
            return false;
        }
    }
    cur = new Node(key);
    //依靠键值比较来决定连接位置(依靠节点地址来判断会误判)
    if (prev->_key > key)
    {
        prev->_left = cur;
    }
    else
    {
        prev->_right = cur;
    }
    return true;
}


  • 递归实现:


bool InsertR(const K& key)
{
    return _InsertR(_root, key);
}
bool _InsertR(Node*& root, const K& key)
{
    if (root == nullptr)
    {
        root = new Node(key);
        return true;
    }
    if (root->_key > key)
    {
        return _InsertR(root->_left, key);
    }
    else if (root->_key < key)
    {
        return _InsertR(root->_right, key);
    }
    else
    {
        return false;
    }
}


5、二叉搜索树的删除



  • 具体操作过程:


若查找元素不存在在二叉搜索树中,则返回false


若查找元素存在,则可能分为下面四种情况:


a. 要删除的结点无孩子结点

b. 要删除的结点只有左孩子结点

c. 要删除的结点只有右孩子结点

d. 要删除的结点有左、右孩子结点

注:实际情况a可以与情况b或者c合并起来


最终的删除过程如下:

情况b:删除该结点且使被删除节点的双亲结点指向被删除节点的左孩子结点

示图:删除91


情况c:删除该结点且使被删除节点的双亲结点指向被删除结点的右孩子结点

示图:删除29


情况d:替换删除,在左子树中找到key最大的(或则在右子树中找到key最小的),与当前结点交换key,然后删除左子树中key最大结点(或则在右子树key最小的结点)

示图:删除20


迭代实现:


bool Erase(const K& key)
{
    Node* cur = _root;
    Node* prev = nullptr;//记录父结点
    while (cur)
    {
        if (cur->_key > key)
        {
            prev = cur;
            cur = cur->_left;
        }
        else if (cur->_key < key)
        {
            prev = cur;
            cur = cur->_right;
        }
        else//找到了
        {
            //分情况讨论
            if (cur->_left == nullptr)
            {
                //根节点特殊处理
                if (cur == _root)
                {
                    _root = cur->_right;
                }
                else
                {
                    //判断是左还是右
                    if (prev->_left == cur)
                    {
                        prev->_left = cur->_right;
                    }
                    else
                    {
                        prev->_right = cur->_right;
                    }
                }
                delete cur;
            }
            else if (cur->_right == nullptr)
            {
                if (cur == _root)
                {
                    _root = cur->_left;
                }
                else
                {
                    if (prev->_left == cur)
                    {
                        prev->_left = cur->_left;
                    }
                    else
                    {
                        prev->_right = cur->_left;
                    }
                }
                delete cur;
            }
            else
            {
                //找到左子树中最大的结点
                Node* maxLeft = cur->_left;
                while (maxLeft->_right)
                {
                    maxLeft = maxLeft->_right;
                }
                K max = maxLeft->_key;
                //先删再替换
                Erase(max);
                cur->_key = max;
            }
            return true;
        }
    }
    return false;
}


  • 递归实现:


bool EraseR(const K& key)
{
    return _EraseR(_root, key);
}
bool _EraseR(Node*& root, const K& key)
{
    if (root == nullptr)
        return false;
    if (root->_key > key)
        return _EraseR(root->_left, key);
    else if (root->_key < key)
        return _EraseR(root->_right, key);
    else
    {
        //记录当前结点
        Node* del = root;
        if (root->_left == nullptr)
        {
            root = root->_right;
        }
        else if (root->_right == nullptr)
        {
            root = root->_left;
        }
        else
        {
            //找到右边最小结点
            Node* minR = root->_right;
            while (minR->_left)
            {
                minR = minR->_left;
            }
            //替换值
            root->_key = minR->_key;
            _EraseR(root->_right, root->_key);
            return true;
        }
        delete del;
        return true;
    }
}


三、二叉搜索树的应用


  1. K模型:


概念:

K模型即只有key作为关键码,结构中只需要存储Key即可,关键码即为需要搜索到的值


示例:给一个单词word,判断该单词是否拼写正确

以单词集合中的每个单词作为key,构建一棵二叉搜索树在二叉搜索树中,检索该单词是否存在,存在则拼写正确,不存在则拼写错误


KV模型:

概念:

每一个关键码key,都有与之对应的值Value,即<Key, Value>的键值


示例:

英汉词典:通过英文可以快速找到与其对应的中文,英文单词与其对应的中文<word, chinese>就构成一种键值对


统计单词次数:统计后,给定单词就可快速找到其出现的次数,单词与其出现次数就是<word, count>就构成一种键值对


实现一个简单的英汉词典dict:

<单词,中文含义>为键值对构造二叉搜索树,二叉搜索树需要比较,键值对比较时只比较Key查询英文单词时,只需给出英文单词,就可快速找到与其对应的key


KV模型:


template<class K, class V>
struct BSTNode
{
  BSTNode(const K& key = K(), const V& value = V())
    : _pLeft(nullptr), _pRight(nullptr), _key(key), _Value(value)
  {}
  BSTNode<T>* _pLeft;
  BSTNode<T>* _pRight;
  K _key;
  V _value
};
template<class K, class V>
class BSTree
{
  typedef BSTNode<K, V> Node;
  typedef Node* PNode;
public:
  BSTree() : _pRoot(nullptr)
  {}
  ~BSTree();
  PNode Find(const K& key);
  bool Insert(const K& key, const V& value)
  {
    // ...
    PNode pCur = _pRoot;
    PNode pParent = nullptr;
    while (pCur)
    {
      pParent = pCur;
      if (key < pCur->_key)
        pCur = pCur->_pLeft;
      else if (key > pCur->_key)
        pCur = pCur->_pRight; // 元素已经在树中存在
      else
        return false;
    }
    // ...
    return true;
  }
  bool Erase(const K& key)
  {
    // ...
    return true;
  }
private:
  PNode _pRoot;
};
相关文章
|
9天前
|
算法 网络安全 区块链
2023/11/10学习记录-C/C++对称分组加密DES
本文介绍了对称分组加密的常见算法(如DES、3DES、AES和国密SM4)及其应用场景,包括文件和视频加密、比特币私钥加密、消息和配置项加密及SSL通信加密。文章还详细展示了如何使用异或实现一个简易的对称加密算法,并通过示例代码演示了DES算法在ECB和CBC模式下的加密和解密过程,以及如何封装DES实现CBC和ECB的PKCS7Padding分块填充。
28 4
2023/11/10学习记录-C/C++对称分组加密DES
|
4月前
|
算法 C语言 C++
C++语言学习指南:从新手到高手,一文带你领略系统编程的巅峰技艺!
【8月更文挑战第22天】C++由Bjarne Stroustrup于1985年创立,凭借卓越性能与灵活性,在系统编程、游戏开发等领域占据重要地位。它继承了C语言的高效性,并引入面向对象编程,使代码更模块化易管理。C++支持基本语法如变量声明与控制结构;通过`iostream`库实现输入输出;利用类与对象实现面向对象编程;提供模板增强代码复用性;具备异常处理机制确保程序健壮性;C++11引入现代化特性简化编程;标准模板库(STL)支持高效编程;多线程支持利用多核优势。虽然学习曲线陡峭,但掌握后可开启高性能编程大门。随着新标准如C++20的发展,C++持续演进,提供更多开发可能性。
92 0
|
1月前
|
存储 C++
【C++】二叉搜索树(BST)
二叉搜索树(Binary Search Tree, BST)是一种特殊的二叉树,其每个节点的左子树所有节点值小于该节点值,右子树所有节点值大于该节点值,且左右子树也均为二叉搜索树。BST支持高效的数据查找、插入和删除操作,时间复杂度通常为O(log n)。本文档详细介绍了BST的基本概念、存储结构及其实现,包括迭代和递归两种方式的查找、插入、删除等操作的具体代码示例。
43 3
|
2月前
|
编译器 C语言 C++
配置C++的学习环境
【10月更文挑战第18天】如果想要学习C++语言,那就需要配置必要的环境和相关的软件,才可以帮助自己更好的掌握语法知识。 一、本地环境设置 如果您想要设置 C++ 语言环境,您需要确保电脑上有以下两款可用的软件,文本编辑器和 C++ 编译器。 二、文本编辑器 通过编辑器创建的文件通常称为源文件,源文件包含程序源代码。 C++ 程序的源文件通常使用扩展名 .cpp、.cp 或 .c。 在开始编程之前,请确保您有一个文本编辑器,且有足够的经验来编写一个计算机程序,然后把它保存在一个文件中,编译并执行它。 Visual Studio Code:虽然它是一个通用的文本编辑器,但它有很多插
|
2月前
|
Java 编译器 C++
c++学习,和友元函数
本文讨论了C++中的友元函数、继承规则、运算符重载以及内存管理的重要性,并提到了指针在C++中的强大功能和使用时需要注意的问题。
30 1
|
5月前
|
存储 C++
【C++】二叉树进阶之二叉搜索树(下)
【C++】二叉树进阶之二叉搜索树(下)
36 4
|
5月前
|
Java 编译器 C++
【C++】二叉树进阶之二叉搜索树(上)
【C++】二叉树进阶之二叉搜索树(上)
40 3
|
5月前
|
存储 安全 编译器
【C++入门 四】学习C++内联函数 | auto关键字 | 基于范围的for循环(C++11) | 指针空值nullptr(C++11)
【C++入门 四】学习C++内联函数 | auto关键字 | 基于范围的for循环(C++11) | 指针空值nullptr(C++11)
|
5月前
|
存储 自然语言处理 编译器
【C++入门 三】学习C++缺省参数 | 函数重载 | 引用
【C++入门 三】学习C++缺省参数 | 函数重载 | 引用
|
29天前
|
存储 编译器 C语言
【c++丨STL】string类的使用
本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
50 2