二叉搜索树 C++代码实现
暂未发现什么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++代码实现的更多相关文章
- python 二叉搜索树相关代码
class TreeNode: def __init__(self, val): self.val = val self.left = None self.right = None class Ope ...
- 基于visual Studio2013解决算法导论之029二叉搜索树
题目 二叉搜索树 解决代码及点评 #include <stdio.h> #include <malloc.h> #include <stdlib.h> ty ...
- 二叉搜索树(Binary Search Tree)--C语言描述(转)
图解二叉搜索树概念 二叉树呢,其实就是链表的一个二维形式,而二叉搜索树,就是一种特殊的二叉树,这种二叉树有个特点:对任意节点而言,左孩子(当然了,存在的话)的值总是小于本身,而右孩子(存在的话)的值总 ...
- 自己动手实现java数据结构(六)二叉搜索树
1.二叉搜索树介绍 前面我们已经介绍过了向量和链表.有序向量可以以二分查找的方式高效的查找特定元素,而缺点是插入删除的效率较低(需要整体移动内部元素):链表的优点在于插入,删除元素时效率较高,但由于不 ...
- 数据结构 - 二叉搜索树封装 C++
二叉搜索树封装代码 #pragma once #include <iostream> using namespace std; template<class T>class T ...
- 每日一题 - 剑指 Offer 33. 二叉搜索树的后序遍历序列
题目信息 时间: 2019-06-26 题目链接:Leetcode tag:分治算法 递归 难易程度:中等 题目描述: 输入一个整数数组,判断该数组是不是某二叉搜索树的后序遍历结果.如果是则返回 tr ...
- 编程算法 - 二叉搜索树 与 双向链表 代码(C++)
二叉搜索树 与 双向链表 代码(C++) 本文地址: http://blog.csdn.net/caroline_wendy 题目:输入一颗二叉搜索树, 将该二叉搜索树转换成一个排序的双向链表. 要求 ...
- 编程算法 - 二叉搜索树(binary search tree) 代码(C)
二叉搜索树(binary search tree) 代码(C) 本文地址: http://blog.csdn.net/caroline_wendy 二叉搜索树(binary search tree)能 ...
- BinarySearchTree(二叉搜索树)原理及C++代码实现
BST是一类用途极广的数据结构.它有如下性质:设x是二叉搜索树内的一个结点.如果y是x左子树中的一个结点,那么y.key<=x.key.如果y是x右子树中的一个结点,那么y.key>=x. ...
随机推荐
- 学习Linux系列--安装Ubuntu
最近学习Linux,使用虚拟机太不方便,于是购买了阿里云最便宜的云主机作为学习设备. 本系列文章记录了个人学习过程的点点滴滴. 学习Linux系列--安装Ubuntu 学习Linux系列--安装软件环 ...
- ASP。net 测验
Login.aspx using System; using System.Collections.Generic; using System.Linq; using System.Web; usin ...
- oracle分析函数
在工作中使用到的分析函数主要有两种,一个是sum () over (partition by ……order by ……)另外一个就是 lead(lag)over (|partition by|ord ...
- Mysql新知识点150928
1.select distinct(DATE_FORMAT(updatetime,'%Y-%m')) as updatetime from barcode where pid!=0 order by ...
- jquery之remove(),detach()方法详解
一:remove()方法 remove()函数用于从文档中移除匹配的元素. 你还可以使用选择器进一步缩小移除的范围,只移除当前匹配元素中符合指定选择器的部分元素. 与detach()相比,remove ...
- 关于实现自定义Dialog和实现Dialog里view的事件监听的两种方法
一.自定义dialog. 二.实现dialog里view的事件监听 1.自定义dialog比较简单.在实例化new的时候,加入样式,布局就行了.或者重写dialog. 2.实现dialog里view的 ...
- C语言细节——献给入门者(一)
C语言细节——献给入门者(一) 主题 输入输出需要注意的细节 首先我们要知道大致有scanf(),printf(),getchar(),putchar(),gets(),puts()这几种输入方式. ...
- CodeForces 166B (凸包)
求一个多边形是否完全在另一个凸多边形内. 乍一看,好像要判点在多边形内,但复杂度不允许,仔细一想,可以把两个多边形的点混起来求一个共同的凸包,如果共同的凸包依旧是原来凸包上的点,说明是. #inclu ...
- Android:padding和android:layout_margin的区别
padding是站在父view的角度描述问题,它规定它里面的内容必须与这个父view边界的距离. margin则是站在自己的角度描述问题,规定自己和其他(上下左右)的view之间的距离
- (转)SQLServer实例讲解
欢迎和大家交流技术相关问题: 邮箱: jiangxinnju@163.com 博客园地址: http://www.cnblogs.com/jiangxinnju GitHub地址: https://g ...