暂未发现什么bug,如果发现请指出。

#include<iostream>
using namespace std;
//定义二叉搜索树的结点
struct Node
{
    int data;
    Node *lc,*rc,*parent;
};
//中序遍历二叉搜索树
void show(Node *now)
{
    if(now==NULL) return;
    show(now->lc);
    cout<<now->data<<endl;
    show(now->rc);
}
//将值为val的结点插入到二叉搜索树中
//?为什么要传指向根结点指针的指针,而不是直接传根结点的指针。因为根结点的指针可能为空,这里需要开辟内存。
Node* insert(Node **rt,int val)
{
    Node *now=*rt;//用now指针在树上移动
    Node *par=NULL;//记录now指针的父亲
    while(now!=NULL)
    {
        par=now;
        if(val<now->data)//如果val小于当前结点的值说明应该插到左子树中
            now=now->lc;
        else
            now=now->rc; //否则插入到右子树中
    }
    Node *newnode=new Node;//开辟新结点
    newnode->data=val;
    newnode->lc=newnode->rc=NULL;
    if(par==NULL)//当这是一棵空树的时候
    {
        newnode->parent=NULL;
        *rt=newnode;//直接通过指针修改根结点
    }
    else
    {
        if(val<par->data)
            par->lc=newnode;
        else
            par->rc=newnode;
        newnode->parent=par;
    }
    return newnode;
}
//在以now为根结点的树上,寻找值为val的结点
Node* search(Node *now,int val)
{
    while(now!=NULL&&now->data!=val)
    {
        if(val<now->data)
            now=now->lc;
        else
            now=now->rc;
    }
    return now;
}
//在以now为根结点的树上,寻找最大、最小值
Node* maximun(Node *now)
{
    while(now->rc!=NULL)
    {
        now=now->rc;
    }
    return now;
}
Node* minimun(Node *now)
{
    while(now->lc!=NULL)
        now=now->lc;
    return now;
}
//寻找now结点的前继、后继
//寻找前继:如果now的左子树非空,则返回左子树的最大值结点。否则如果now是父亲的左孩子则不断向上直到now不再是父亲的左孩子,返回父结点。
Node* predecessor(Node *now)
{
    if(now->lc!=NULL)
        return maximun(now->lc);
    Node *par=now->parent;
    while(par!=NULL&&now==par->lc)
    {
        now=par;
        par=par->parent;
    }
    return par;
}
Node* successor(Node *now)
{
    if(now->rc!=NULL)
        return minimun(now->rc);
    Node *par=now->parent;
    while(par!=NULL&&now==par->rc)
    {
        now=par;
        par=par->parent;
    }
    return par;
}
//对于一棵以T为根结点的树,用以v为根结点的子树取代以u为根结点的子树,完成u的父亲与v之间的绑定
void transplant(Node **T,Node *u,Node *v)
{
    if(u->parent==NULL)
        (*T)=v;
    else if(u==u->parent->lc)
        u->parent->lc=v;
    else if(u==u->parent->rc)
        u->parent->rc=v;
    if(v!=NULL)
        v->parent=u->parent;
}
//删除给定结点
void erase(Node **T,Node *now)
{
    Node *temp=now;
    if(now->lc==NULL)
    {
        transplant(T,now,now->rc);
    }
    else if(now->rc==NULL)
    {
        transplant(T,now,now->lc);
    }
    else
    {
        Node* nextnode=successor(now);
        if(nextnode!=now->rc)
        {
            transplant(T,nextnode,nextnode->rc);
            nextnode->rc=now->rc;
            now->rc->parent=nextnode;
        }
        transplant(T,now,nextnode);

        nextnode->lc=now->lc;
        now->lc->parent=nextnode;
    }
    delete temp;
}
int main()
{
    Node *root=NULL;
    )
    {
        cout<<"1.插入结点"<<endl;
        cout<<"2.删除结点"<<endl;
        cout<<"3.中序遍历"<<endl;
        cout<<"4.查询"<<endl;
        cout<<"5.查询根"<<endl;
        int p;
        cin>>p;
        )
        {
            cout<<"输入待插元素值:"<<endl;
            int t;
            cin>>t;
            insert(&root,t);
            cout<<"插入成功!"<<endl;
        }
        )
        {
            cout<<"输入待删元素值:"<<endl;
            int t;
            cin>>t;
            Node *q=search(root,t);
            if(q==NULL)cout<<"该值不存在!"<<endl;
            else
            {
                erase(&root,q);
                cout<<"删除成功!"<<endl;
            }
        }
        )
        {
            cout<<"======"<<endl;
            show(root);
            cout<<"======"<<endl;
        }
        )
        {
            cout<<"输入待查询元素值:"<<endl;
            int t;
            cin>>t;
            Node *q=search(root,t);
            if(q==NULL) cout<<"该值不存在!"<<endl;
            else
            {
                Node *a=predecessor(q),*b=successor(q);
                if(a!=NULL)
                    cout<<"前继:"<<a->data<<endl;
                if(b!=NULL)
                    cout<<"后继:"<<b->data<<endl;
            }
        }
        )
        {
            if(root==NULL) cout<<"根为空!"<<endl;
            else cout<<"根:"<<root->data<<endl;
        }
    }
    ;
}

二叉搜索树 C++代码实现的更多相关文章

  1. python 二叉搜索树相关代码

    class TreeNode: def __init__(self, val): self.val = val self.left = None self.right = None class Ope ...

  2. 基于visual Studio2013解决算法导论之029二叉搜索树

     题目 二叉搜索树 解决代码及点评 #include <stdio.h> #include <malloc.h> #include <stdlib.h> ty ...

  3. 二叉搜索树(Binary Search Tree)--C语言描述(转)

    图解二叉搜索树概念 二叉树呢,其实就是链表的一个二维形式,而二叉搜索树,就是一种特殊的二叉树,这种二叉树有个特点:对任意节点而言,左孩子(当然了,存在的话)的值总是小于本身,而右孩子(存在的话)的值总 ...

  4. 自己动手实现java数据结构(六)二叉搜索树

    1.二叉搜索树介绍 前面我们已经介绍过了向量和链表.有序向量可以以二分查找的方式高效的查找特定元素,而缺点是插入删除的效率较低(需要整体移动内部元素):链表的优点在于插入,删除元素时效率较高,但由于不 ...

  5. 数据结构 - 二叉搜索树封装 C++

    二叉搜索树封装代码 #pragma once #include <iostream> using namespace std; template<class T>class T ...

  6. 每日一题 - 剑指 Offer 33. 二叉搜索树的后序遍历序列

    题目信息 时间: 2019-06-26 题目链接:Leetcode tag:分治算法 递归 难易程度:中等 题目描述: 输入一个整数数组,判断该数组是不是某二叉搜索树的后序遍历结果.如果是则返回 tr ...

  7. 编程算法 - 二叉搜索树 与 双向链表 代码(C++)

    二叉搜索树 与 双向链表 代码(C++) 本文地址: http://blog.csdn.net/caroline_wendy 题目:输入一颗二叉搜索树, 将该二叉搜索树转换成一个排序的双向链表. 要求 ...

  8. 编程算法 - 二叉搜索树(binary search tree) 代码(C)

    二叉搜索树(binary search tree) 代码(C) 本文地址: http://blog.csdn.net/caroline_wendy 二叉搜索树(binary search tree)能 ...

  9. BinarySearchTree(二叉搜索树)原理及C++代码实现

    BST是一类用途极广的数据结构.它有如下性质:设x是二叉搜索树内的一个结点.如果y是x左子树中的一个结点,那么y.key<=x.key.如果y是x右子树中的一个结点,那么y.key>=x. ...

随机推荐

  1. aaaaaaaaaaaaaaa

    <?xml version="1.0" encoding="utf-8"?><document> <!-- 签名,由平台生成 -- ...

  2. 登录锁定状态下Win7关机技巧总结

    登录锁定状态下Win7关机技巧总结 一般在锁定状态都是有个关闭电脑的图标的.但是如果你的系统没有,那么怎么样关机呢,所谓的锁定状态通常是指电脑在登录界面,具体的实现如下,感兴趣的朋友可以参考下 现在大 ...

  3. [Python爬虫]cnblogs博客备份工具(可扩展成并行)

    并发爬虫小练习. 直接粘贴到本地,命名为.py文件即可运行,运行时的参数为你想要爬取的用户.默认是本博客. 输出是以用户名命名的目录,目录内便是博客内容. 仅供学习python的多线程编程方法,后续会 ...

  4. Kanzi UI Solution

    Kanzi UI Solution是一个完整的跨平台的UI解决方案, 基于OpenGL 和 OpenGL ES.Kanzi为UI的设计.开发和部署在嵌入式设备上的图形用户界面提供一个完善的开发平台. ...

  5. [bzoj3155]Preprefix sum(树状数组)

    3155: Preprefix sum Time Limit: 1 Sec  Memory Limit: 512 MBSubmit: 1183  Solved: 546[Submit][Status] ...

  6. HTML基础(2)

    常见标签 1.img标签 <img src="图片地址" alt="图片名"/>  图片(单标签) alt属性 是图片名字,是给百度搜索引擎抓取使用 ...

  7. UIDatePicker的用法

    目录[-] 1.Locale 2.Calendar 3.timeZone 4.date 5.minimumDate 6.maximumDate 7.countDownDuration 8.minute ...

  8. ASP.NET MVC电子书下载

    欢迎点击图片下载电子书,进行深入的研究.                (  ,)      

  9. es6箭头函数中this

    普通函数: $scope.$on('$stateChangeSuccess',function(){this.list = this.getList();}); 箭头函数: $scope.$on('$ ...

  10. 转<%%>、<%=%>、<%$%>、<%@%>的区别

    1. 未定义的命名空间前缀“xsd” 上周在项目开发中遇到这样的一个问题,在一个页面用到了自定义的Picker控件,在IE6.7.8.9以及IE10兼容模式下都没有任何问题,但是一换到IE10时已选择 ...