二叉搜索树的创建和删除

一、二叉搜索树的概念
二叉搜索树又称为二叉排序树,它或者是一棵空树,或者是具有以下性质的二叉树:
1.若它的左子树不为空,则左子树上所有结点的值都小于根结点的值。
2.若它的右子树不为空,则右子树上所有结点的值都大于根结点的值。
3.它的左右子树也分别是二叉搜索树。

下面就是一颗二叉搜索树
在这里插入图片描述
二、二叉搜索树的算法实现
1、创建二叉搜索树
创建二叉搜索树的方法就是将每一个结点依次插入二叉搜索树。主要思想是为目标节点找出合适的叶节点位置,然后将该节点作为叶节点插入。
在这里插入图片描述
在这里插入图片描述
在左边这个树里插入4,从根节点开始,4小于5,因此继续搜索左子树。4大于2,而且2没有右孩子,因此4作为2的右孩子插入到树中。

2、删除二叉搜索树的一个结点
这里的删除操作是用一个合适的子节点来替换要删除的目标节点,使整体操作变化最小。如果目标节点值大于当前节点,则指向右子树去寻找目标节点;如果目标节点值小于当前节点,则指向左子树去寻找目标节点。如果找到了目标节点,我们分为三种情况:

1:目标节点没有子节点,可以直接删除目标节点。如下图所示:
在这里插入图片描述

2:目标节点有右结点,右子树最左边的结点进行替代(也就是右子树里面最小的结点),该结点位于右子树中较低的位置。然后可以从后继结点的位置递归向下操作以删除后继结点。如下图所示:
在这里插入图片描述

3:目标节点只有左节点,可以使用它的左子树最右边的节点进行替代(也就是左子树里面的最大节点),然后再递归的向下删除节点。如下图所示:
在这里插入图片描述

#include<iostream>
#include<queue>
using namespace::std;
typedef struct Node
{
	int data;
	Node* lchild, * rchild;
}Btree;


//插入一个结点
Btree* InsertNode(Btree* T, int k)
{
	if (T == nullptr)
	{
		T = new Btree;
		T->data = k;
		T->lchild = NULL;
		T->rchild = NULL;
		return T;
	}
	if (k > T->data)
	{
		T->rchild=InsertNode(T->rchild, k);
	}
	else
	{
		T->lchild=InsertNode(T->lchild, k);
	}
	return T;
}

Btree* CreatTree(Btree* T, int a[],int len)
{
	
	for (int i = 0; i < len; i++)
	{
		T=InsertNode(T, a[i]);   //将数据插入二叉搜索树中
	}
	return T;

}
//先序遍历
void PreOrder(Btree* t)
{
	if (t)
	{
		cout << t->data<<" ";     //输出结点数据
		PreOrder(t->lchild);//访问左孩子
		PreOrder(t->rchild);//访问右孩子
	}
}
Btree* DeleteNode(Btree* T, int key)
{
	Btree* Node = T;
	Btree* parent=new Btree;
	int flag=-1;   
	//找到要删除的结点
	while (Node) {
		if (Node->data == key)
		{
			break;
		}
		else if (key > Node->data)  //如果要寻找的值大于结点的值,则向右子树遍历
		{
			parent = Node;  //保存父节点
			flag = 1;   //表明删除的结点是父节点右孩子
			Node = Node->rchild;
		}
		else
		{
			parent = Node;  //保存父节点
			flag = 0;   //表明删除的结点是父结点左孩子
			Node = Node->lchild;    //如果要寻找的值小于结点的值,则向左子树遍历
		}
	}
	//要删除的结点无孩子结点,直接删除
	if (Node->lchild == NULL && Node->rchild == NULL)
	{
		if (flag == -1) //如果只有T一个结点,直接删除,返回一个空指针
		{
			return NULL;
		}
		else if (flag == 0)//删除左孩子
		{
			parent->lchild = NULL;
		}
		else if (flag == 1)//删除右孩子
		{
			parent->rchild = NULL;
		}
	}
	//要删除的结点有右孩子
	else if (Node->rchild != NULL)
	{
		Btree* MostLchild = Node->rchild;
		int temp=MostLchild->data;
		//找到右孩子中的最左边的结点,也就是右孩子中最小的结点
		while (MostLchild->lchild)
		{
			MostLchild = MostLchild->lchild;
			temp = MostLchild->data;  //保存这个结点的值
		}
		//先删除再交换,如果先交换那么二叉树里面就有两个相同的结点了
		DeleteNode(T, MostLchild->data);  //删除右孩子的最小的结点,这一步就是相当于将要删除的结点与右孩子的最小结点互换了,然后将右孩子的最小结点删除
		Node->data = temp;  //将右子树中最小的结点的值覆盖掉要删除的结点的值
	}
	//要删除的结点没有右孩子,只有左孩子
	else
	{
		Btree* MostRchild = Node->lchild;
		int temp;
		//找到左孩子中的最右边的结点,也就是左孩子中最大的结点
		while (MostRchild->rchild)
		{
			MostRchild = MostRchild->rchild;
			temp = MostRchild->data;   //保存这个结点的值
		}
		DeleteNode(T, MostRchild->data);
		Node->data = temp; //将左子树中最大的结点的值覆盖掉要删除的结点的值
	}
	return T;
}
int main()
{
	int a[10] = { 5,3,7,1,4,6,8,0,2,9 };
	Btree* T=NULL;
	Btree* BT=CreatTree(T, a,10);   //创建二叉搜索树
	cout << "先序遍历:";
	PreOrder(BT);   //先序遍历二叉树
	cout << endl;
	BT=DeleteNode(BT,8);   //删除值为8的结点
	cout << "删除后:";
	PreOrder(BT);   //先序遍历二叉树
}

运行结果:
在这里插入图片描述

  • 10
    点赞
  • 69
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
设二叉排序树的二叉链表存储结构的类型定义如下: typedef struct node{ int data; //用整数表示一个结点的名 struct node *LChild,*RChild; //左右指针域 }BSTNode,*BSTree; 设计算法并编写程序求解以下几个问题。 8 12 14 10 7 3 15 6 2 4 1 5 11 9 13 16 13 (1)键盘输入一个元素序列创建一棵二叉排序树,输出该二叉排序树的中序遍历序列; 例如,若输入 45,24,55,12,37,53,60,23,40,70 则创建的二叉排序树为: 输出结果为:12 23 24 37 40 45 53 55 60 70 (2)在(1)中所得的二叉排序树中插入一个值为 58 的结点,再输出它的中序遍历序列,输出 结果为:12 23 24 37 40 45 53 55 58 60 70 (3)在(1)中所得的二叉排序树中删除值为 45 的结点,再输出它的中序遍历序列,输出结果 为:12 23 24 37 40 53 55 58 60 70 (4)利用(1)中所得的二叉排序树的所有叶子结点构造一个带头结点的单链表 L。要求不能 破坏这棵二叉排序树。所得的单链表 L 如下。 输出该链表各结点的值,输出结果为:23 40 53 70 (5)设计算法将(1)中所得的二叉排序树的左右子树进行交换,由于二叉树是一种递归定义, 所以子树的左右两棵子树也要相交换,依此类推。最后输出所得到的二叉树的中序遍历序列。 例如,经过上述操作后,(1)中所得的二叉排序树变为如下形式。 输出该二叉树的中序序列,结果为:70 60 55 53 45 40 37 24 23 12 (6)设计算法统计并输出(1)中所得的二叉排序树中只有一个孩子结点的结点个数。输出结 果为:3(7)在(1)中所得的二叉排序树中,设计算法并编写程序输出结点 40 的所有祖先结点。输 出结果为:45 24 37

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值