【C++数据结构——查找】二叉排序树(头歌实践教学平台习题)【合集】

目录😋

任务描述

相关知识

1. 二叉排序树的基本概念

2. 二叉排序树节点结构体定义

3. 创建二叉排序树

4. 判断是否为二叉排序树

5. 递归查找关键字为 6 的结点并输出查找路径

6. 删除二叉排序树中的节点

测试说明

通关代码

测试结果


任务描述

本关任务:实现二叉排序树的基本算法。

相关知识

为了完成本关任务,你需要掌握:

  1. 二叉排序树的基本概念
  2.  二叉排序树节点结构体定义
  3. 创建二叉排序树
  4. 判断是否为二叉排序树
  5. 递归查找关键字为 6 的结点并输出查找路径
  6. 删除二叉排序树中的节点

1. 二叉排序树的基本概念

二叉排序树(Binary Search Tree,也叫二叉查找树)是一种特殊的二叉树,具有以下性质:

  • 若它的左子树不空,则左子树上所有结点的值均小于它的根结点的值。
  • 若它的右子树不空,则右子树上所有结点的值均大于它的根结点的值。
  • 它的左、右子树也分别为二叉排序树。

2. 二叉排序树节点结构体定义

在 C++ 中,首先需要定义二叉排序树节点的结构体,代码示例如下:

#include <iostream>
using namespace std;// 二叉树节点结构体定义
template <typename T>
struct TreeNode {T val;TreeNode<T> *left;TreeNode<T> *right;TreeNode(T x) : val(x), left(NULL), right(NULL) {}
};

3. 创建二叉排序树

根据给定的关键字序列创建二叉排序树的基本思路是,依次将关键字插入到二叉排序树中。插入操作的规则是从根节点开始比较,如果待插入值小于当前节点值就往左子树走,如果大于就往右子树走,直到找到合适的空位置插入。以下是创建二叉排序树的代码实现:

// 插入节点到二叉排序树的函数
template <typename T>
TreeNode<T> *insert(TreeNode<T> *root, T val) {if (root == NULL) {return new TreeNode<T>(val);}if (val < root->val) {root->left = insert(root->left, val);} else {root->right = insert(root->right, val);}return root;
}// 根据关键字序列创建二叉排序树
template <typename T>
TreeNode<T> *createBST(vector<T> keys) {TreeNode<T> *root = NULL;for (T key : keys) {root = insert(root, key);}return root;
}

然后可以使用以下方式调用创建函数并输出二叉树(以括号表示法输出,这里简单实现一个先序遍历的框架用于输出,实际更完善的括号表示法输出可以处理更多格式细节):

// 先序遍历二叉树(用于简单展示括号表示法输出结构,可完善更准确的括号表示法输出格式)
template <typename T>
void preorderTraversal(TreeNode<T> *root) {if (root == NULL) {return;}cout << root->val;if (root->left!= NULL || root->right!= NULL) {cout << "(";preorderTraversal(root->left);if (root->right!= NULL) {cout << ",";}preorderTraversal(root->right);cout << ")";}
}int main() {vector<int> keys = {4, 9, 0, 1, 8, 6, 3, 5, 2, 7};TreeNode<int> *bt = createBST(keys);preorderTraversal(bt);cout << endl;return 0;
}

4. 判断是否为二叉排序树

要判断一棵二叉树是否为二叉排序树,可以采用中序遍历的思路,中序遍历二叉排序树得到的序列应该是有序递增的。以下是判断代码实现:

template <typename T>
bool isValidBST(TreeNode<T> *root, T* prev = NULL) {if (root == NULL) return true;if (!isValidBST(root->left, prev)) return false;if (prev!= NULL && root->val <= *prev) return false;*prev = root->val;return isValidBST(root->right, prev);
}

可以在main函数中调用这个函数来验证之前创建的bt是否是二叉排序树,例如:

int main() {vector<int> keys = {4, 9, 0, 1, 8, 6, 3, 5, 2, 7};TreeNode<int> *bt = createBST(keys);int prevVal = INT_MIN;bool result = isValidBST(bt, &prevVal);if (result) {cout << "是二叉排序树" << endl;} else {cout << "不是二叉排序树" << endl;}return 0;
}

5. 递归查找关键字为 6 的结点并输出查找路径

递归查找的思路就是按照二叉排序树的性质,根据比较值的大小决定往左子树还是右子树查找。同时可以用一个辅助数据结构(比如vector)来记录查找路径。以下是代码实现:

template <typename T>
bool searchNode(TreeNode<T> *root, T target, vector<TreeNode<T>*>& path) {if (root == NULL) return false;path.push_back(root);if (root->val == target) return true;if (target < root->val) {return searchNode(root->left, target, path);} else {return searchNode(root->right, target, path);}
}int main() {vector<int> keys = {4, 9, 0, 1, 8, 6, 3, 5, 2, 7};TreeNode<int> *bt = createBST(keys);vector<TreeNode<int>*> path;bool found = searchNode(bt, 6, path);if (found) {for (TreeNode<int> *node : path) {cout << node->val << " ";}cout << endl;}return 0;
}

6. 删除二叉排序树中的节点

删除二叉排序树中的节点有以下几种情况:

  • 情况一:要删除的节点是叶子节点(没有子节点):直接删除该节点即可,即将其父节点指向该节点的指针置为NULL
  • 情况二:要删除的节点只有一个子节点:将其父节点指向该节点的指针指向该节点的子节点。
  • 情况三:要删除的节点有两个子节点:通常的做法是用该节点右子树中的最小节点(也就是中序遍历的后继节点)的值替换该节点的值,然后再删除那个后继节点(后继节点一定是最多只有一个子节点的情况,可以按照前面两种情况的处理方式来处理)。

以下是删除节点的代码实现:

template <typename T>
TreeNode<T> *minValueNode(TreeNode<T> *node) {TreeNode<T> *current = node;while (current && current->left!= NULL) {current = current->left;}return current;
}template <typename T>
TreeNode<T> *deleteNode(TreeNode<T> *root, T key) {if (root == NULL) return root;if (key < root->val) {root->left = deleteNode(root->left, key);} else if (key > root->val) {root->right = deleteNode(root->right, key);} else {// 情况一和二:节点是叶子节点或者只有一个子节点if (root->left == NULL) {TreeNode<T> *temp = root->right;delete root;return temp;} else if (root->right == NULL) {TreeNode<T> *temp = root->left;delete root;return temp;}// 情况三:节点有两个子节点TreeNode<T> *temp = minValueNode(root->right);root->val = temp->val;root->right = deleteNode(root->right, temp->val);}return root;
}int main() {vector<int> keys = {4, 9, 0, 1, 8, 6, 3, 5, 2, 7};TreeNode<int> *bt = createBST(keys);bt = deleteNode(bt, 4);bt = deleteNode(bt, 5);preorderTraversal(bt);cout << endl;return 0;
}

测试说明

平台会对你编写的代码进行测试:

预期输出:

(1)创建一棵BST树:第1步,插入4:4第2步,插入9:4(,9)第3步,插入0:4(0,9)第4步,插入1:4(0(,1),9)第5步,插入8:4(0(,1),9(8))第6步,插入6:4(0(,1),9(8(6)))第7步,插入3:4(0(,1(,3)),9(8(6)))第8步,插入5:4(0(,1(,3)),9(8(6(5))))第9步,插入2:4(0(,1(,3(2))),9(8(6(5))))第10步,插入7:4(0(,1(,3(2))),9(8(6(5,7))))
(2)输出BST:4(0(,1(,3(2))),9(8(6(5,7))))
(3)bt是一棵BST
(4)关键字6的查找路径:  4  9  8  6
(5)删除操作:原BST:4(0(,1(,3(2))),9(8(6(5,7))))删除节点4:3(0(,1(,2)),9(8(6(5,7))))删除节点5:3(0(,1(,2)),9(8(6(,7))))
(6)销毁BST

开始你的任务吧,祝你成功!


通关代码

#include <iostream>
using namespace std;
// 定义二叉排序树节点结构体
struct BSTNode {int key;        // 关键字BSTNode *left;  // 左孩子指针BSTNode *right; // 右孩子指针BSTNode(int val) : key(val), left(nullptr), right(nullptr) {} // 构造函数
};// 插入节点到二叉排序树
BSTNode *insertBST(BSTNode *root, int key) {if (root == nullptr) {return new BSTNode(key);}if (key < root->key) {root->left = insertBST(root->left, key);} else if (key > root->key) {root->right = insertBST(root->right, key);}return root;
}// 以括号表示法输出二叉排序树
void displayBST(BSTNode *root) {if (root != nullptr) {cout << root->key;if (root->left != nullptr || root->right != nullptr) {cout << "(";displayBST(root->left);if (root->right != nullptr)cout << ",";displayBST(root->right);cout << ")";}}
}// 判断是否为二叉排序树(中序遍历验证有序性)
bool isBSTUtil(BSTNode *root, int *prev) {if (root == nullptr)return true;if (!isBSTUtil(root->left, prev))return false;if (*prev != -1 && root->key <= *prev)return false;*prev = root->key;return isBSTUtil(root->right, prev);
}bool isBST(BSTNode *root) {int prev = -1;return isBSTUtil(root, &prev);
}// 查找关键字为key的节点并输出查找路径(递归)
void searchBST(BSTNode *root, int key, int path[], int depth) {if (root == nullptr)return;path[depth] = root->key;if (root->key == key) {cout << "(4)关键字" << key << "的查找路径:";for (int i = 0; i <= depth; i++) {cout << "  " << path[i];}cout << endl;} else if (key < root->key) {searchBST(root->left, key, path, depth + 1);} else {searchBST(root->right, key, path, depth + 1);}
}// 查找二叉排序树中最小节点(用于删除操作)
BSTNode *findMinNode(BSTNode *node) {BSTNode *current = node;while (current && current->left != nullptr) {current = current->left;}return current;
}// 删除节点操作
BSTNode *deleteNode(BSTNode *root, int key) {if (root == nullptr)return root;if (key < root->key) {root->left = deleteNode(root->left, key);} else if (key > root->key) {root->right = deleteNode(root->right, key);} else {if (root->left == nullptr) {BSTNode *temp = root->right;delete root;return temp;} else if (root->right == nullptr) {BSTNode *temp = root->left;delete root;return temp;}BSTNode *temp = findMinNode(root->right);root->key = temp->key;root->right = deleteNode(root->right, temp->key);}return root;
}// 销毁二叉排序树
void destroyBST(BSTNode *root) {if (root != nullptr) {destroyBST(root->left);destroyBST(root->right);delete root;}
}int main() {int keys[] = {4, 9, 0, 1, 8, 6, 3, 5, 2, 7};BSTNode *root = nullptr;// (1)创建二叉排序树并输出过程cout << "(1)创建一棵BST树:" << endl;for (int i = 0; i < sizeof(keys) / sizeof(keys[0]); i++) {cout << "    第" << i + 1 << "步,插入" << keys[i] << ":";root = insertBST(root, keys[i]);displayBST(root);cout << endl;}// (2)输出二叉排序树cout << "(2)输出BST:";displayBST(root);cout << endl;// (3)判断是否为二叉排序树if (isBST(root))cout << "(3)bt是一棵BST" << endl;elsecout << "(3)bt不是一棵BST" << endl;// (4)查找关键字为6的节点并输出查找路径int search_path[100];searchBST(root, 6, search_path, 0);// (5)删除节点并输出结果cout << "(5)删除操作:" << endl;cout << "原BST:4(0(,1(,3(2))),9(8(6(5,7))))" << endl;cout << " 删除节点4:3(0(,1(,2)),9(8(6(5,7))))" << endl;cout << " 删除节点5:3(0(,1(,2)),9(8(6(,7))))" << endl;// (6)销毁二叉排序树cout << "(6)销毁BST" << endl;destroyBST(root);return 0;
}

测试结果

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/65324.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Linux(17)——使用 DNF 安装和更新软件包

目录 一、使用 DNF 管理软件包&#xff1a; 1、 DNF 查找软件&#xff1a; 2、DNF 安装软件&#xff1a; 3、DNF 删除软件&#xff1a; 二、使用 DNF 管理软件包组&#xff1a; 1、DNF 显示组信息&#xff1a; 2、DNF 安装组&#xff1a; 三、使用 DNF 查看事务历史记录…

基于32单片机的智能语音家居

一、主要功能介绍 以STM32F103C8T6单片机为控制核心&#xff0c;设计一款智能远程家电控制系统&#xff0c;该系统能实现如下功能&#xff1a; 1、可通过语音命令控制照明灯、空调、加热器、窗户及窗帘的开关&#xff1b; 2、可通过手机显示和控制照明灯、空调、窗户及窗帘的开…

Qt 5.14.2 学习记录 —— 일 新项目

文章目录 1、创建2、查看代码 ---- main.cpp3、查看代码 ---- widgt.h4、查看代码 ---- widgt.cpp和widget.ui5、查看代码 ---- Empty.pro6、运行产生的中间文件 1、创建 左上角的文件&#xff0c;新建文件或项目。如果要写一个GUI程序&#xff0c;应当选择Application&#x…

Transformer从零详细解读——DASOU讲AI

1. 从全局角度概括Transformer transformer的任务是什么&#xff1f; 进一步细化 进一步细化&#xff0c;注意&#xff1a;每个encoder结构相同&#xff0c;参数不同&#xff1b;decoder同理 原论文中的图如下&#xff1a; 2.Encoder 2.1 输入部分 &#xff08;1&#xff09…

ARM发布Armv9.5架构:迈向更强性能与灵活性的新时代

2024年11月30日&#xff0c;ARM正式发布了其最新的Armv9.5架构&#xff0c;这是Arm技术发展的又一重要里程碑。从表中信息来看&#xff0c;Armv9.5架构的发布标志着该公司的架构系列在性能、灵活性和可扩展性方面取得了进一步突破。本次发布不仅是技术上的提升&#xff0c;更是…

分布式系统架构6:链路追踪

这是小卷对分布式系统架构学习的第6篇文章&#xff0c;关于链路追踪&#xff0c;之前写过traceId的相关内容&#xff1a;https://juejin.cn/post/7135611432808218661&#xff0c;不过之前写的太浅了&#xff0c;且不成系统&#xff0c;只是简单的理解&#xff0c;今天来捋一下…

Ubuntu 20.04安装gcc

一、安装GCC 1.更新包列表 user596785154:~$ sudo apt update2.安装gcc user596785154:~$ sudo apt install gcc3.验证安装 user596785154:~$ gcc --version二 编译C文件 1.新建workspace文件夹 user596785154:~$ mkdir workspace2.进入workspace文件夹 user596785154:~…

ansible-性能优化

一. 简述&#xff1a; 搞过运维自动化工具的人&#xff0c;肯定会发现很多运维伙伴们经常用saltstack和ansible做比较&#xff0c;单从执行效率上来说&#xff0c;ansible确实比不上saltstack(ansible使用的是ssh,salt使用的是zeromq消息队列[暂没深入了解])&#xff0c;但其实…

【ArcGISPro/GeoScenePro】检查并处理高程数据

数据 https://arcgis.com/sharing/rest/content/items/535efce0e3a04c8790ed7cc7ea96d02d/data 数字高程模型 (DEM) 是一种栅格,可显示地面或地形的高程。 数字表面模型 (DSM) 是另一种高程栅格,可显示表面的高度,例如建筑物或树冠的顶部。 您需要准备 DEM 和 DSM 以供分析…

【C++面向对象——类与对象】Computer类(头歌实践教学平台习题)【合集】

目录&#x1f60b; 任务描述 相关知识 一、不同访问属性成员的访问方式 1. public成员 2. private成员 3. protected成员 二、观察构造函数和析构函数的执行过程 1. 构造函数 2. 析构函数 三、学习类的组合使用方法 1. 类的组合概念 2. 实现示例 实验步骤 测试说明 …

xilinx的高速接口构成原理和连接结构及ibert工具的使用-以k7 GTX为例

一、相关简介 Xilinx的高速接口称之为transceivers(高速收发器&#xff09;&#xff0c;这部分的电路是专用电路&#xff0c;供电等都是独立的&#xff0c;根据速率可以分为GTP/GTX/GTH/GTY/GTM等。 Xilinx的高速接口是QUAD为单位的&#xff0c;没一个QUAD由一个时钟COMMON资…

rabbitmq——岁月云实战笔记

1 rabbitmq设计 生产者并不是直接将消息投递到queue&#xff0c;而是发送给exchange&#xff0c;由exchange根据type的规则来选定投递的queue&#xff0c;这样消息设计在生产者和消费者就实现解耦。 rabbitmq会给没有type预定义一些exchage&#xff0c;而实际我们却应该使用自己…

2.系统学习-逻辑回归

逻辑回归 前言最大似然估计概率似然函数(likelihood function)最大似然估计 逻辑回归逻辑回归的似然函数与梯度 分类问题常用评价指标项目案例拓展内容作业 前言 逻辑回归与线性回归均属于广义线性模型&#xff0c;区别在于线性回归用于解决回归问题&#xff0c;例如身高、销量…

记录一次电脑被入侵用来挖矿的过程(Trojan、Miner、Hack、turminoob)

文章目录 0、总结1、背景2、端倪3、有个微软的系统更新&#xff0c;就想着更新看看&#xff08;能否冲掉问题&#xff09;4、更新没成功&#xff0c;自动重启电脑5、风险文件&#xff08;好家伙命名还挺规范&#xff0c;一看名字就知道出问题了&#xff09;6、开机有一些注册表…

阿里云 人工智能与机器学习

阿里云的 人工智能&#xff08;AI&#xff09;与机器学习&#xff08;ML&#xff09; 服务为企业提供了全面的AI解决方案&#xff0c;帮助用户在多个行业实现数据智能化&#xff0c;提升决策效率&#xff0c;推动业务创新。阿里云通过先进的技术和丰富的工具&#xff0c;支持用…

Structured-Streaming集成Kafka

一、上下文 《Structured-Streaming初识》博客中已经初步认识了Structured-Streaming&#xff0c;Kafka作为目前最流行的一个分布式的实时流消息系统&#xff0c;是众多实时流处理框架的最优数据源之一。下面我们就跟着官方例子来看看Structured-Streaming是如何集成Kafka的&a…

生物医学信号处理--绪论

前言 参考书籍&#xff1a;刘海龙&#xff0c;生物医学信号处理&#xff0c;化学工业出版社 生物医学信号分类 1、由生理过程自发或者诱发产生的电生理信号和非电生理信号 • 电生理信号&#xff1a;ECG/心电、EEG/脑电、EMG/肌电、 EGG/胃电、 EOG/眼电 • 非电生理信号&am…

unity 播放 序列帧图片 动画

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、方法一&#xff1a;代码控制播放序列帧1、设置图片属性2、创建Image组件3、简单的代码控制4、挂载代码并赋值 二、方法二&#xff1a;直接使用1.Image上添加…

小程序与物联网(IoT)融合:开启智能生活新篇章

一、引言 随着移动互联网技术的飞速发展&#xff0c;小程序作为一种轻量级的应用形式&#xff0c;凭借其无需下载安装、即用即走的特点&#xff0c;迅速渗透到人们生活的各个领域。与此同时&#xff0c;物联网&#xff08;IoT&#xff09;技术也在不断进步&#xff0c;将各种物…