【C++高阶】高效搜索的秘密:深入解析搜索二叉树

本文涉及的产品
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
简介: 【C++高阶】高效搜索的秘密:深入解析搜索二叉树

前言: 在数据结构和算法的广阔领域中,二叉搜索树(Binary Search Tree,简称BST)无疑是一颗璀璨的明星。它以其高效的数据检索能力和独特的树形结构,在计算机科学领域扮演着举足轻重的角色。对于任何对编程和数据结构感兴趣的人来说,掌握二叉搜索树都是至关重要的一步

二叉搜索树不仅仅是一个简单的数据结构,它更是一种解决问题的方式和思维的体现。通过维护二叉树中每个节点的左子树所有值均小于它的值,右子树所有值均大于它的值的特性,二叉搜索树在插入、查找和删除操作中展现出了卓越的性能。这种特性使得二叉搜索树在各种应用中成为了一种理想的数据结构选择,从基础的算法练习到复杂的系统优化,都能见到它的身影

学习二叉搜索树并非易事。它需要我们深入理解其性质、原理和算法实现。我们需要掌握如何构建一棵二叉搜索树,如何遍历它,以及如何在其中进行高效的查找、插入和删除操作。这些都需要我们付出大量的时间和精力去学习和实践。我们将从二叉搜索树的基本概念出发,逐步深入到其性质、构建、遍历以及操作的实现

让我们一起踏上学习二叉搜索树的旅程,探索它带来的无尽可能!
(本文重在二叉搜索树的模拟实现与理解)


📒1. 二叉搜索树

🎩二叉搜索树概念

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

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


🎈二叉搜索树操作

首先,在二叉搜索树的操作中只支持插入,查找,删除,遍历,并不支持修改操作,因为在修改后谁也不能保证它依然是一棵二叉搜索树,二叉搜索树的时间复杂度范围在(O(logN)~O(N))

在二叉搜索树的遍历中一般采用中序遍历: 先遍历左子树,然后访问根节点,最后遍历右子树。在BST中,中序遍历会按照升序访问所有节点

二叉搜索树示例

int a[] = {8, 3, 1, 10, 6, 4, 7, 14, 13};

📕2. 二叉搜索树模拟实现

🧩二叉搜索树结构

二叉搜索树结构的和树形结构差不多,这意味着每个元素(通常称为节点)都有两个指针:一个指向前一个左子树,另一个指向右子树,因此我们需要单独再定义一个类来表示节点结构,每个节点再串联起来构成BST

(在模拟实现二叉搜索树时,不用定义命名空间,因为不会和库中发生冲突)

节点定义(示例):

template<class K>
struct BSTreeNode
{
  BSTreeNode<K>* _left;
  BSTreeNode<K>* _right;
  K _key;
  BSTreeNode(const K& key = K())
    :_key(key)
    , _left(nullptr)
    , _right(nullptr)
  {}
};

BST定义(示例):

template<class K>
class BSTree
{
  typedef BSTreeNode<K> Node;
public:
  // 构造函数等可能的其他成员函数... 
private:
  Node* _root = nullptr;
};

🧩二叉搜索树操作

🌈插入

插入的具体过程如下:

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

插入代码实现(示例):

bool Insert(const K& key)
{
  // 当根为空时,直接插入
  if (_root == nullptr)
  {
    _root = new Node(key);
    return true;
  }
  // 定义parent是因为,在最后找到插入位置时,需要parent将节点进行连接
  Node* parent = nullptr; 
  Node* cur = _root;
  while (cur)
  {
    parent = cur;
    // 插入的值比cur位置大时,cur往右移动
    if (key > cur->_key)
    {
      cur = cur->_right;
    }
    // 插入的值比cur位置小时,cur往左移动
    else if (key < cur->_key)
    {
      cur = cur->_left;
    }
    // 当插入的值和cur位置相等时,直接退出,因为二叉搜索树不允许有相同的元素
    else
    {
      return false;
    }
  }
  // 将插入位置的新节点与二叉搜索树连接
  cur = new Node(key);
  if (parent->_key < key)
  {
    parent->_right = cur;
  }
  else
  {
    parent->_left = cur;
  }
  return true;
}

🌞遍历

在二叉搜索树的遍历上,我们依旧采用当初二叉树时的中序遍历,但是我们想要递归遍历就必须调用节点,这里我们要调用两层

遍历代码实现(示例):

void InOrder()
{
  _InOrder(_root);
  cout << endl;
}
void _InOrder(Node* root)
{
  // 递归出口
  if (root == nullptr)
  {
    return;
  }
  _InOrder(root->_left);
  cout << root->_key << " " ;
  _InOrder(root->_right);
}

遍历比较简单奥,我们接着往下讲


🌙查找

二叉搜索树的查找

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

查找代码实现(示例):

bool Find(const K& key)
{
  Node* cur = _root;
  while (cur)
  {
    // 查找的值比cur大,cur往右移动
    if (key > cur->_key)
    {
      cur = cur->_right;
    }
    // 查找的值比cur小,cur往左移动
    else if (key < cur->_key)
    {
      cur = cur->_left;
    }
    // 找到key,返回true
    else
    {
      return true;
    }
  }
  return false; // 找不到key,返回false
}

⭐删除

首先查找元素是否在二叉搜索树中,如果不存在,则返回, 否则要删除的结点可能分下面情况:

  • 要删除的结点无孩子结点
  • 要删除的结点只有左孩子结点
  • 要删除的结点只有右孩子结点
  • 要删除的结点有左、右孩子结点

而上面四种情况可以转化成下面的情况:

  • 删除该结点且使被删除节点的双亲结点指向被删除节点的左孩子结点–直接删除
  • 删除该结点且使被删除节点的双亲结点指向被删除结点的右孩子结点–直接删除
  • 在它的右子树中寻找中序下的第一个结点(关键码最小),用它的值填补到被删除节点中,再来处理该结点的删除问题–替换法删除

这三种情况就是我们模拟实现删除的方法!

删除代码实现(示例):

bool Erase(const K& key)
{
  Node* cur = _root;
  Node* parent = nullptr;
  while (cur)
  {
    parent = cur;
    if (key > cur->_key)
    {
      cur = cur->_right;
    }
    else if (key < cur->_key)
    {
      cur = cur->_left;
    }
    else
    {
      // 准备删除
      // 左为空
      if (cur->_left == nullptr)
      {
        // 当需要删除的是头节点时
        if (cur == _root)
        {
          _root = cur->_right;
        }
        else
        {
          if (cur == parent->_left)
          {
            parent->_left = cur->_right;
          }
          else
          {
            parent->_right = cur->_right;
          }
        }
      }
      // 右为空
      else if (cur->_right == nullptr)
      {
        // 当需要删除的是头节点时
        if (cur == _root)
        {
          _root = cur->_left;
        }
        else
        {
          if (cur == parent->_left)
          {
            parent->_left = cur->_left;
          }
          else
          {
            parent->_right = cur->_left;
          }
        }
      }
      // 两边都不为空
      else
      {
        // 这里与外层不是同一块作用域,所以可以再次定义parent,不把parent定义为nullptr是因为
        //,可能不进入下面循环导致对parent空指针的使用
        Node* subLeft = cur->_right; // 定义右数节点
        Node* parent = cur;
        while (subLeft->_left)
        {
          parent = subLeft;
          subLeft = subLeft->_left;
        }
        swap(cur->_key, subLeft->_key); // 替换右子树的最左节点
        if (subLeft == parent->_left)
        {
          // 最左节点肯定没有左孩子,所以让parent接管它的右子树
          parent->_left = subLeft->_right;
        }
        else
        {
          parent->_right = subLeft->_right;
        }
        delete subLeft;
      }
      return true;
    }
  }
  return false;
}

🧩二叉搜索树默认成员函数

构造

BSTree() = default; // 显式地定义默认构造函数  

拷贝构造

BSTree(const BSTree<K>& t)
{
  _root = Copy(t._root);
}
Node* Copy(Node* root)
{
  if (root == nullptr)
  {
    return nullptr;
  }
  // 递归进行拷贝构造
  Node* newroot = new Node(root->_key);
  newroot->_left = Copy(root->_left);
  newroot->_right = Copy(root->_right);
  
  return newroot;
}

赋值重载

BSTree<K>& operator=(BSTree<K> t)
{
  // 现代写法-> 直接调用swap
  swap(_root, t._root);
  return *this;
}

析构

~BSTree()
{
  Destory(_root);
}
void Destory(Node*& _root)
{
  if (_root == nullptr)
  {
    return;
  }
  // 递归调用析构
  Destory(_root->_left);
  Destory(_root->_right);
  delete _root;
  
    _root == nullptr;
}

📜3. 二叉搜索树模拟实现(递归)

在进行递归操作的模拟实现时,一般都要传节点,进行多层的调用,因为我们都要定义两层

bool FindR(const K& key)
{
  return _FindR(_root, key);
}
bool InsertR(const K& key)
{
  return _InsertR(_root, key);
}
bool EraseR(const K& key)
{
  return _EraseR(_root, key);
}

🌞插入

代码实现(示例):

bool _InsertR(Node*& _root, const K& key)
{
  // 递归出口
  if (_root == nullptr)
  {
    // 这里我们无需在进行对新节点的连接,因为我们是传引用传参,
    _root = new Node(key);
    return true;
  }
  if (key > _root->_key)
  {
    return _InsertR(_root->_right, key);
  }
  else if (key < _root->_key)
  {
    return _InsertR(_root->_left, key);
  }
  else
  {
    return false;
  }
}

🌙查找

代码实现(示例):

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

⭐删除

代码实现(示例):

bool _EraseR(Node*& _root, const K& key)
{
  if (_root == nullptr)
  {
    return false;
  }
  if (key > _root->_key)
  {
    return _EraseR(_root->_right, key);
  }
  else if (key < _root->_key)
  {
    return _EraseR(_root->_left, key);
  }
  else
  {
    // 删除
    if (_root->_left == nullptr)
    {
      Node* del = _root;
      _root = _root->_right;
      delete del;
      return true;
    }
    else if (_root->_right == nullptr)
    {
      Node* del = _root;
      _root = _root->_left;
      delete del;
      return true;
    }
    else
    {
      Node* subLeft = _root->_right;
      while (subLeft->_left)
      {
        subLeft = subLeft->_left;
      }
    swap(_root->_key, subLeft->_key);
    // 让子树继续递归下去
    return _EraseR(_root->_right, key);
    }
  }
}

📚4. 二叉搜索树的应用

🍁KV模型

KV模型:每一个关键码key,都有与之对应的值Value,即<Key, Value>的键值对。该种方式在现实生活中非常常见:

  • 比如英汉词典就是英文与中文的对应关系,通过英文可以快速找到与其对应的中文,英
    文单词与其对应的中文<word, chinese>就构成一种键值对
  • 再比如统计单词次数,统计成功后,给定单词就可快速找到其出现的次数,单词与其出
    现次数就是<word, count>就构成一种键值对
namespace kv // 避免与之前k模型冲突
{
  template<class K, class V>
  struct BSTreeNode
  {
    BSTreeNode<K>* _left;
    BSTreeNode<K>* _right;
    K _key;
    V _value;
    
    BSTreeNode(const K& key = K(), const V& value = V())
      : _left(nullptr)
      , _right(nullptr)
      , _key(key)
      , _value(value)
    {}
  };
  template<class K, class V>
  class BSTree
  {
    typedef BSTreeNode<K, V> Node;
  public:
    // 构造函数等可能的其他成员函数... 
    // 在成员函数中,我们只需要在insert中加入value元素即可
  private:
    Node* _root = nullptr;
  };
}

在成员函数中,我们只需要在Insert中加入value元素即可

🍂KV模型实现

💧英汉词典

代码实现(示例):

int main()
{
  kv::BSTree<string, string> dict;
  dict.insert("left", "左边、剩余");
  dict.insert("string", "字符串");
  dict.insert("hahaha", "哈哈");
  dict.insert("Eternity", "永恒");
  dict.insert("sort", "排序");
  dict.InOrder();
  string str;
  while (cin >> str)
  {
    kv::BSTreeNode<string, string>* ret = dict.Find(str);
    if (ret)
    {
      cout << ret->_value << endl;
    }
    else
    {
      cout << "无此单词" << endl;
    }
  }
}

🔥计数

代码实现(示例):

int main()
{
  string arr[] = { "苹果", "西瓜", "苹果", "西瓜", "苹果", "苹果", "西瓜",
"苹果", "香蕉", "苹果", "香蕉" };
  kv::BSTree<string, int> countTree;
  for (auto& e : arr)
  {
    kv::BSTreeNode<string, int>* ret = countTree.Find(e);
    if (ret == nullptr)
    {
      countTree.insert(e, 1);
    }
    else
    {
      ret->_value++;
    }
  }
  countTree.InOrder();
  return 0;
}

🌄二叉树巩固知识

最后在这给大家推荐几道巩固二叉树的编程题

二叉树创建字符串

二叉树的分层遍历

找到树中两个指定节点的最近公共祖先

二叉树搜索树转换成排序双向链表

根据一棵树的前序遍历与中序遍历构造二叉树

二叉树中序遍历 ,非递归迭代实现


📖5. 总结

经过我们一同对搜索二叉树的深入学习和探索,相信你已经对这种数据结构有了全面而深刻的理解。搜索二叉树以其独特的性质在数据检索领域展现了出色的性能,无论是插入、删除还是查找操作,都体现了其高效和灵活的特点

学习的道路永无止境。虽然我们已经走过了搜索二叉树的基本概念和操作的学习之旅,但这只是你编程旅程中的一个站点。前方还有更多数据结构等待你去探索,更多算法等待你去实践

不要忘记持续学习和自我提升。计算机科学是一个日新月异的领域,新的技术和算法不断涌现。只有保持对知识的渴望和追求,我们才能在这个领域中不断前行,让我们一起在学习的道路上不断前行!

希望本文能够为你提供有益的参考和启示,让我们一起在编程的道路上不断前行!

目录
相关文章
|
20天前
|
Java C++
【C++数据结构——树】二叉树的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现二叉树的基本运算。​ 相关知识 创建二叉树 销毁二叉树 查找结点 求二叉树的高度 输出二叉树 //二叉树节点结构体定义 structTreeNode{ intval; TreeNode*left; TreeNode*right; TreeNode(intx):val(x),left(NULL),right(NULL){} }; 创建二叉树 //创建二叉树函数(简单示例,手动构建) TreeNode*create
40 12
|
20天前
|
C++
【C++数据结构——树】二叉树的性质(头歌实践教学平台习题)【合集】
本文档介绍了如何根据二叉树的括号表示串创建二叉树,并计算其结点个数、叶子结点个数、某结点的层次和二叉树的宽度。主要内容包括: 1. **定义二叉树节点结构体**:定义了包含节点值、左子节点指针和右子节点指针的结构体。 2. **实现构建二叉树的函数**:通过解析括号表示串,递归地构建二叉树的各个节点及其子树。 3. **使用示例**:展示了如何调用 `buildTree` 函数构建二叉树并进行简单验证。 4. **计算二叉树属性**: - 计算二叉树节点个数。 - 计算二叉树叶子节点个数。 - 计算某节点的层次。 - 计算二叉树的宽度。 最后,提供了测试说明及通关代
39 10
|
20天前
|
存储 算法 测试技术
【C++数据结构——树】二叉树的遍历算法(头歌教学实验平台习题) 【合集】
本任务旨在实现二叉树的遍历,包括先序、中序、后序和层次遍历。首先介绍了二叉树的基本概念与结构定义,并通过C++代码示例展示了如何定义二叉树节点及构建二叉树。接着详细讲解了四种遍历方法的递归实现逻辑,以及层次遍历中队列的应用。最后提供了测试用例和预期输出,确保代码正确性。通过这些内容,帮助读者理解并掌握二叉树遍历的核心思想与实现技巧。
40 2
|
28天前
|
存储 算法 安全
基于红黑树的局域网上网行为控制C++ 算法解析
在当今网络环境中,局域网上网行为控制对企业和学校至关重要。本文探讨了一种基于红黑树数据结构的高效算法,用于管理用户的上网行为,如IP地址、上网时长、访问网站类别和流量使用情况。通过红黑树的自平衡特性,确保了高效的查找、插入和删除操作。文中提供了C++代码示例,展示了如何实现该算法,并强调其在网络管理中的应用价值。
|
1月前
|
安全 编译器 C++
C++ `noexcept` 关键字的深入解析
`noexcept` 关键字在 C++ 中用于指示函数不会抛出异常,有助于编译器优化和提高程序的可靠性。它可以减少代码大小、提高执行效率,并增强程序的稳定性和可预测性。`noexcept` 还可以影响函数重载和模板特化的决策。使用时需谨慎,确保函数确实不会抛出异常,否则可能导致程序崩溃。通过合理使用 `noexcept`,开发者可以编写出更高效、更可靠的 C++ 代码。
44 1
|
1月前
|
存储 程序员 C++
深入解析C++中的函数指针与`typedef`的妙用
本文深入解析了C++中的函数指针及其与`typedef`的结合使用。通过图示和代码示例,详细介绍了函数指针的基本概念、声明和使用方法,并展示了如何利用`typedef`简化复杂的函数指针声明,提升代码的可读性和可维护性。
92 1
|
1月前
|
机器学习/深度学习 搜索推荐 API
淘宝/天猫按图搜索(拍立淘)API的深度解析与应用实践
在数字化时代,电商行业迅速发展,个性化、便捷性和高效性成为消费者新需求。淘宝/天猫推出的拍立淘API,利用图像识别技术,提供精准的购物搜索体验。本文深入探讨其原理、优势、应用场景及实现方法,助力电商技术和用户体验提升。
|
2月前
|
自然语言处理 编译器 Linux
|
1月前
|
数据采集 XML 数据格式
解析Amazon搜索结果页面:使用BeautifulSoup
解析Amazon搜索结果页面:使用BeautifulSoup
|
2月前
|
设计模式 安全 数据库连接
【C++11】包装器:深入解析与实现技巧
本文深入探讨了C++中包装器的定义、实现方式及其应用。包装器通过封装底层细节,提供更简洁、易用的接口,常用于资源管理、接口封装和类型安全。文章详细介绍了使用RAII、智能指针、模板等技术实现包装器的方法,并通过多个案例分析展示了其在实际开发中的应用。最后,讨论了性能优化策略,帮助开发者编写高效、可靠的C++代码。
54 2

热门文章

最新文章

推荐镜像

更多