数据结构-考研难点代码突破(C++实现树型查找-二叉搜索树(二叉排序树))

news2024/11/27 20:27:14

文章目录

  • 1.二叉搜索树基本操作
    • 二叉搜索树的效率分析
  • 2. C++实现

1.二叉搜索树基本操作

二叉排序树是具有下列特性的二叉树:

  1. 若左子树非空,则左子树上所有结点的值均小于根结点的值。
  2. 若右子树非空,则右子树上所有结点的值均大于根结点的值。
  3. 左、右子树也分别是一棵二叉排序树

根据二叉排序树的定义,左子树结点值< 根结点值< 右子树结点值。
所以对二叉排序树进行中序遍历,可以得到一个递增的有序序列。

二叉排序树的搜索

  1. 从根结点开始,沿某个分支逐层向下比较的过程。
  2. 若二叉排序树非空,先将给定值与根结点的关键字比较,若相等,则查找成功
  3. 若不等,如果小于根结点的关键字,则在根结点的左子树上查找,否则在根结点的右子树上查找。

这个过程可以通过递归或者非递归实现实现

二叉排序树的插入

  1. 若原二叉排序树为空,则直接插入结点
  2. 若关键字k小于根结点值,则插入到左子树,若关键字k大于根结点值,则插入到右子树。

插入的结点一定是一个新添加的叶结点,且是查找失败时的查找路径上访问的最后一个结点的左孩子或右孩子.

二叉排序树的删除

在二叉排序树中删除一个结点时,不能把以该结点为根的子树上的结点都删除,必须先把被删除结点从存储二叉排序树的链表上摘下,将因删除结点而断开的二叉链表重新链接起来,同时确保二叉排序树的性质不会丢失。删除操作的实现过程按3种情况来处理

  1. 若被删除结点z是叶结点,则直接删除,不会破坏二叉排序树的性质

  2. 若结点z只有一棵左子树或右子树,则让z的子树成为z父结点的子树,替代z的位置

  3. 若结点z有左、右两棵子树,则令z的直接后继(或直接前驱)替代z,然后从二叉排序树中删去这个直接后继(或直接前驱),这样就转换成了第一或第二种情况。可以使用递归解决这个问题。

    直接后继:节点右子树中最左节点(右子树最小节点)

    删除流程图:具体看代码流程
    在这里插入图片描述

二叉搜索树的效率分析

二叉排序树的查找效率,主要取决于树的高度。

  1. 若二叉排序树的左、右子树的高度之差的绝对值不超过1(这样的二叉排序树称为平衡二叉树)它的平均查找长度为O(logN)

  2. 若二叉排序树是一个只有右(左)孩子的单支树(类似于有序的单链表),则其平均查找长度为O(N)

在这里插入图片描述
a图的平均搜索成功长度ASL:类似折半查找(1x1 + 2x2+……h×2h-1) )(每层的节点不一定放满,需要针对题目灵活处理)
(1×1+2×2+3×4+4×3)÷10=2.9

b图的平均搜索成功长度ASL:(1+10)/2=5.5
因为此时搜索二叉树由树型退化为线型,平均搜索长度为(n+1)/2

二叉排序树与二分查找相似。就平均时间性能而言,二叉排序树上的查找和二分查找差不多。

但二分查找的判定树唯一,而二叉排序树的查找不唯一,相同的关键字其插入顺序不同可能生成不同的二叉排序树。

二叉排序树与二分搜索:

  1. 二叉排序树无须移动结点,只需修改指针即可完成插入和删除操作, 平均执行时间为O(logN)
  2. 二分查找的对象是有序顺序表,若有插入和删除结点的操作,所花的代价是O(n)。
  3. 当有序表是静态查找表时,宜用顺序表作为其存储结构,采用二分查找实现其查找操作;若有序表是动态查找表,则应选择二叉排序树作为其逻辑结构.

2. C++实现

// 二叉搜索树头文件实现
#include <iostream>
#include <vector>
#include <algorithm>

struct TreeNode
{
    int _val;
    TreeNode *_left;
    TreeNode *_right;
    TreeNode(int val) : _val(val), _left(nullptr), _right(nullptr) {}
};

class SearchTree
{
private:
    TreeNode *root = nullptr;
    /**
     * @brief 在二叉搜索树中查找节点值为val的节点
     *
     * @param val 查找节点的值
     * @param node 如果找到了这个节点,node就是这个节点的地址,否则为null
     * @param part part这个节点的父亲,如果这个节点为null,这个参数输出null
     */
    void _find(int val, TreeNode *&node, TreeNode *&part)
    {
        TreeNode *ptr = root;
        TreeNode *prev = nullptr;
        while (ptr != nullptr)
        {
            if (ptr->_val == val)
            {
                node = ptr;
                break;
            }
            else if (ptr->_val > val)
            {
                prev = ptr;
                ptr = ptr->_left;
            }
            else
            {
                prev = ptr;
                ptr = ptr->_right;
            }
        }
        node = ptr;
        part = prev;
    }

    bool _erase(int val, TreeNode *&node)
    {
        if (node == nullptr)
        {
            return false;
        }
        else
        {
            if (node->_val > val)
            {
                _erase(val, node->_left);
            }
            else if (node->_val < val)
            {
                _erase(val, node->_right);
            }
            else
            {

                if (node->_left == nullptr)
                {
                    TreeNode *del = node;
                    node = node->_right;
                    delete del;
                }
                else if (node->_right == nullptr)
                {
                    TreeNode *del = node;
                    node = node->_left;
                    delete del;
                }
                else
                {
                    // 找要删除节点的后继
                    TreeNode *right_min_node = node->_right;
                    while (right_min_node->_left != nullptr)
                    {
                        right_min_node = right_min_node->_left;
                    }

                    // 记录right_min_node这个节点的值,吧这个节点的值和node节点交换,在删除right_min_node这个节点即可
                    // right_min_node这个节点的左子树一定为空,在上面顶点流程会处理
                    int tmp = right_min_node->_val;
                    _erase(tmp, node->_right);
                    node->_val = tmp;
                }
            }
            return true;
        }
    }

    // 判断一棵树是否是二叉搜索树,检查每个节点是否满足节点值的取值范围
    bool _isSearchTree(TreeNode *node, int min, int max)
    {
        if (node == nullptr)
            return true;

        if (node->_val < min || node->_val > max)
        {
            return false;
        }
        return _isSearchTree(node->_left, min, node->_val - 1) && _isSearchTree(node->_right, node->_val + 1, max);
    }

    void _display_inorder(TreeNode *node)
    {
        if (node == nullptr)
            return;
        _display_inorder(node->_left);
        std::cout << node->_val << " ";
        _display_inorder(node->_right);
    }

    int _max()
    {
        TreeNode *node = root;
        while (node->_right != nullptr)
        {
            node = node->_right;
        }
        return node->_val;
    }

    int _min()
    {
        TreeNode *node = root;
        while (node->_left != nullptr)
        {
            node = node->_left;
        }
        return node->_val;
    }

public:
    SearchTree() = default;
    SearchTree(const std::vector<int> &buff)
    {
        for (int i = 0; i < buff.size(); i++)
        {
            insert(buff[i]);
        }
    }
    // 不允许重复插入相同的值
    bool insert(int val)
    {
        if (root == nullptr)
        {
            root = new TreeNode(val);
            return true;
        }
        else
        {
            TreeNode *pos = nullptr;
            TreeNode *prev = nullptr;
            _find(val, pos, prev);
            if (pos != nullptr)
            {
                // 之前插入过,值重复
                return false;
            }
            else
            {
                // pos==nullptr;
                if (prev->_val > val)
                {
                    prev->_left = new TreeNode(val);
                }
                else
                {
                    prev->_right = new TreeNode(val);
                }
                return true;
            }
        }
    }

    // 查找节点值为val的节点
    bool find(int val)
    {
        TreeNode *node;
        TreeNode *part;
        _find(val, node, part);
        return node != nullptr;
    }

    // 删除搜索二叉树节点
    bool erase(int val)
    {
        return _erase(val, root);
    }

    // 判断这颗树是否为二叉搜索树
    bool isSearchTree()
    {
        return _isSearchTree(root, _min(), _max());
    }

    // 中序遍历二叉搜索树
    void inorder()
    {
        _display_inorder(root);
        std::cout << "\n";
    }
};

测试代码:

#include "SearchTree.h"
#include <time.h>

using namespace std;

#define TIMES 10

int main(int argc, char const *argv[])
{
    srand((unsigned int)time(0));
    // vector<int> ret = {1, 2, 3};
    vector<int> ret;
    for (int i = 0; i < TIMES; i++)
    {
        ret.push_back(rand() % 100);
    }
    SearchTree tree(ret);
    tree.inorder();
    tree.insert(40);
    tree.inorder();
    cout << tree.isSearchTree() << endl;
    tree.insert(1);
    tree.erase(40);
    tree.inorder();
    cout << tree.isSearchTree() << endl;
    return 0;
}

在这里插入图片描述

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

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

相关文章

蓝牙耳机哪个品牌性价比高?性价比高的无线蓝牙耳机

现如今耳机已经十分普及&#xff0c;大多数人会随身佩戴蓝牙耳机&#xff0c;相较于传统耳机&#xff0c;无线耳机不仅携带方便&#xff0c;舒适度上也更加出色。不过市面上的无线耳机种类繁多&#xff0c;很多朋友不知道该如何挑选&#xff0c;所以小编特意整理了一期性价比高…

从零开始搭建kubernetes集群环境(虚拟机/kubeadm方式)

文章目录1 Kubernetes简介&#xff08;k8s&#xff09;2 安装实战2.1 主机安装并初始化2.2 安装docker2.3 安装Kubernetes组件2.4 准备集群镜像2.5 集群初始化2.6 安装flannel网络插件3 部署nginx 测试3.1 创建一个nginx服务3.2 暴漏端口3.3 查看服务3.4 测试服务1 Kubernetes简…

2023如何选购适合游戏设计的电脑硬件

游戏设计涉及许多不同的学科&#xff0c;因此涉及许多不同的软件包。有游戏引擎本身&#xff0c;例如 Unreal Engine 和 Unity&#xff0c;以及 3D 设计软件&#xff0c;例如 3ds Max、Blender 和 ZBrush——等等&#xff01;大多数软件开发人员都维护着这些不同应用程序的系统…

C++设计模式(14)——享元模式

亦称&#xff1a; 缓存、Cache、Flyweight 意图 享元模式是一种结构型设计模式&#xff0c; 它摒弃了在每个对象中保存所有数据的方式&#xff0c; 通过共享多个对象所共有的相同状态&#xff0c; 让你能在有限的内存容量中载入更多对象。 问题 假如你希望在长时间工作后放…

U-Mail邮件系统安全攻略之邮件监控

随着互联网的普及&#xff0c;电子邮件已经成为现代人生活和工作中最常用的通信工具之一&#xff0c;在企业的内外文件往来和日常沟通中发挥着重要作用。电子邮件在获得广泛应用的同时&#xff0c;也让企业的重要信息安全受到很大的威胁&#xff0c;比如员工未经允许将公司重要…

复杂美公链技术重要特色:平行公链架构

复杂美公链技术Chain33从11月开源至今&#xff0c;获得众多合作方的认可&#xff0c;其中首创的平行公链架构被百度、阿里、360等机构认可并跟进研究&#xff0c;这也说明了平行公链或许是区块链普及应用的重要解决方案之一。 平行公链&#xff08;以下简称平行链&#xff09;是…

【ADRC控制】使用自抗扰控制器调节起动机入口压力值

以前只知道工业控制中用的是PID控制&#xff0c;然而最近了解到实际生产中还在使用ADRC控制&#xff0c;而且使用效果还优于PID控制&#xff0c;遂找了几篇文献学习学习。 0 引言 自抗扰控制&#xff08;Active Disturbances Rejection Controller&#xff0c;ADRC&#xff09;…

【Datawhale图机器学习】第一章图机器学习导论

图机器学习导论 学习路径与必读论文清单 斯坦福CS224W&#xff08;子豪兄中文精讲&#xff09;知识图谱实战DeepwalkNode2vecPageRankGNNGCNGragh-SAGEGINGATTrans-ETrans-R 图无处不在 图是描述关联数据的通用语言 举例 计算机网络新冠肺炎流行病学调查传播链食物链地铁图…

推荐一个.Ner Core开发的配置中心开源项目

更多开源项目请查看&#xff1a;一个专注推荐.Net开源项目的榜单 当你把单体应用改造为微服务架构&#xff0c;相应的配置文件&#xff0c;也会被分割&#xff0c;被分散到各个节点。这个时候就会产生一个问题&#xff0c;配置信息是分散的、冗余的&#xff0c;变成不好维护管理…

ZigBee案例笔记 - USART

文章目录1.串行通信接口简述2.串行通信接口寄存器U0CSR (0x86) -USART 0 控制和状态U0UCR (0xC4)–USART 0 UART 控制U0GCR (0xC5)–USART 0 通用控制U0BUF (0xC1) – USART 0 接收/传送数据缓存U0BAUD (0xC2) – USART 0 波特率控制3.设置串行通信接口比特率控制寄存器4.外设I…

分布式一致性算法Raft原理图释

什么是分布式一致性算法Raft 分布式一致性算法Raft&#xff1a;指在分布式场景下实现集群数据同步的解决方案 掌握了这个算法&#xff0c;就可以较容易地处理绝大部分场景的容错和数据一致性需求 Raft三大角色 跟随者&#xff08;Follower&#xff09;&#xff1a;普通群众…

opencv绘制矩形和圆

大家好&#xff0c;我是csdn的博主&#xff1a;lqj_本人 这是我的个人博客主页&#xff1a; lqj_本人的博客_CSDN博客-微信小程序,前端,python领域博主lqj_本人擅长微信小程序,前端,python,等方面的知识https://blog.csdn.net/lbcyllqj?spm1011.2415.3001.5343哔哩哔哩欢迎关注…

Go语言之 下载安装go以及vscode配置go环境

上篇请移步到Go语言之 下载安装及第一个代码_水w的博客-CSDN博客 目录 一、下载安装以及配置go环境 1 下载安装go 2 配置go环境 二、安装配置git 一、在vscode上开发golang 1 配置 2 编写代码 解决报错&#xff1a;go: go.mod file not found in current directory or …

J-Link RTT Viewer使用教程(附代码)

目录 RTT(Real Time Transfer)简介 使用教程 常用API介绍 RTT缓冲大小修改 使用printf重定向 官方例程 RTT(Real Time Transfer)简介 平常调试代码中使用串口打印log&#xff0c;往往需要接出串口引脚&#xff0c;比较麻烦&#xff0c;并且串口打印速度较慢&#xff0c;串…

RTT 线程间同步互斥

1.概念 在多线程实时系统中&#xff0c;一项工作的完成往往可以通过多个线程协调的方式共同来完成&#xff0c;那么多个线程之间如何 “默契” 协作才能使这项工作无差错执行&#xff1f;下面举个例子说明。 例如一项工作中的两个线程&#xff1a;一个线程从传感器中接收数据…

分享112个HTML电子商务模板,总有一款适合您

分享112个HTML电子商务模板&#xff0c;总有一款适合您 112个HTML电子商务模板下载链接&#xff1a;https://pan.baidu.com/s/13wf9C9NtaJz67ZqwQyo74w?pwdzt4a 提取码&#xff1a;zt4a Python采集代码下载链接&#xff1a;采集代码.zip - 蓝奏云 有机蔬菜水果食品商城网…

ubuntu 安装支持GPU的Docker详细步骤

安装依赖项 sudo apt-get update sudo apt-get install -y apt-transport-https ca-certificates curl gnupg-agent software-properties-common 添加 Docker GPG 密钥 curl -fsSL https://download.docker.com/linux/ubuntu/gpg | sudo apt-key add - sudo apt-key fingerpr…

分享111个HTML医疗保健模板,总有一款适合您

分享111个HTML医疗保健模板&#xff0c;总有一款适合您 111个HTML医疗保健模板下载链接&#xff1a;https://pan.baidu.com/s/1YInaQDnUVsXYtMh1Ls-BHg?pwdxvfc 提取码&#xff1a;xvfc Python采集代码下载链接&#xff1a;采集代码.zip - 蓝奏云 import os import shuti…

元宇宙之声:新鸿基公司

在本期节目中&#xff0c;新鸿基团队讲述了他们在农历新年季中展示的元宇宙最新创作&#xff01; 为什么将体验命名为「乘风启航」&#xff1f;什么是 「Scallywag」&#xff1f; 香港专业离岸帆船队新鸿基 Scallywag 队由新鸿基公司赞助&#xff0c;其团队精神与公司的精神相呼…

Git实用指令记录

config 用例&#xff1a;对git最先要做的一个操作就是配置用户名和邮箱&#xff0c;否则无法commit查看所有可以config的条目&#xff0c;非常之多$ git config --list core.symlinksfalse core.autocrlftrue core.fscachetrue color.interactivetrue color.uiauto help.forma…