二叉排序树的C++实现

    最近复习了一下C++语言,复习的过程中顺带重温了下数据结构的一些内容,一石二鸟呀!下面是类模板实现二叉排序树。

    先定义节点类型和二叉树类型:

template<typename T>
class BinaryTree;

template<typename T>   //节点类的定义
class Node
{private:
	Node<T>*lchild, *rchild;
	T info;
 public:
	 Node()
	 {
		 lchild = NULL;
		 rchild = NULL;
	 }
	 Node(T data, Node<T>*left = NULL, Node<T>*right = NULL)
	 {
		 info = data;
		 lchild = left;
		 rchild = right;
	 }
	 friend class BinaryTree < T > ;
};

template<typename T>                 //二叉排序树类的定义     
class BinaryTree
{
	Node<T>*root;
	void Inorder(Node<T>*current);              
	void Insert(const T &data,Node<T>* &b);
	void Remove(const T &data, Node<T>* &a, Node<T>* &b);
	void Destory(Node<T>*current);                           //删除二叉排序树
public:
	BinaryTree(){ root = NULL; }
	~BinaryTree(){ Destory(root); }
	void Create(T*data, int n);                      //由数组创建二叉排序树,n是数组大小
	void InOrder(){ Inorder(root); }                  //中序遍历
	void Remove(const T&data){ Remove(data,root,root); }  //删除节点
};

          上面定义的二叉排序树类中,涉及的一些简单操作:

template<typename T>
void BinaryTree<T>::Insert(const T &data, Node<T>* &b)  //递归法在二叉排序树中插入一个节点
{
	if (b == NULL)          //递归终止条件
	{
		b = new Node<T>(data);
		if (!b)
		{
			cout << "out of space!" << endl;
		}
	}
	else if (data < b->info)
		Insert(data, b->lchild);
	else
		Insert(data, b->rchild);
}

template<typename T>
void BinaryTree<T>::Inorder(Node<T>*current)  //中序遍历
{
	if (current != NULL)
	{
		Inorder(current->lchild);
		cout << current->info<<" ";
		Inorder(current->rchild);
	}
}

template<typename T>
void BinaryTree<T>::Create(T*data, int n)         //由数组创建二叉排序树,n是数组大小
{
	for (int i = 0; i < n; i++)
	{
		Insert(data[i],root);
	}
}

     关于二叉排序树的有关操作,个人认为,最有技术含量的是节点的删除。这个问题需要考虑的情况较多点。

删除一个节点需要考虑对应节点的状态,具体的说就是,节点是否为空,若不空,它的左右子树是否存在。我们可以按以下步骤来进行:

     一、查找待删除节点,在查找的同时需要记录一下待删除节点的父亲,因为删除节点的同时需要善后(将其父节点的lchild或者rchild置空)。

     二、根据待删除节点的左右子树状态做相应处理,这些状态包括:

           1.如果待删除节点的左右节点都不存在,那么直接删除,将其父节点相应的指针置空(如果存在父节点)。

           2.如果待删除节点左子树存在右子树不存在,或者左子树不存在右子树存在。直接将其子树中存在的一边候补上来即可。

           3.如果待删除节点左右子树都在,这个情况稍微复杂点。需要按照二叉排序树的性质从其左子树或者有子树中选择节点补到待删除节点的位置。

             一般是用树的中序遍历中跟待删除节点相邻的节点(左右节点都行,这里就选右节点值)来替带待删除的节点。实际上执行删除操作时,并不

是真的把待删除节点删掉,而是用替代节点的键值赋值给待删节点键值,然后delete掉替代节点。

                                                                 

                                    (a)                                                                                                                                                 (b)

                上面从图(a)到(b)是一个典型待删节点左右子树都存在的删除过程:要删除15节点,先把21节点改为20节点的左子树,再把18节点见值复值给待删除的15节点,最后删除原18节点。

template<typename T>
void BinaryTree<T>::Remove(const T &data, Node<T>* &a, Node<T>* &b)
{
	         Node<T>*temp1, *temp2;
		if (b == NULL) { cerr << "Invalid input"; return; }
		if (data < b->info) Remove(data, b, b->lchild);    //先找到节点位置,这里没有考虑找不到的情况
		else if (data >b->info) Remove(data, b, b->rchild);

                else if (b->lchild != NULL&&b->rchild != NULL)     //这个分支以及下面的分支表示已找到data所在节点
		{
			temp2 = b;
			temp1 = b->rchild;
			if (temp1->lchild != NULL)
			{
				while (temp1->lchild != NULL)
				{
					temp2 = temp1;
					temp1 = temp1->lchild;
				}
				temp2->lchild = temp1->rchild;
			}
			else temp2->rchild = temp1->rchild;
			b->info = temp1->info;
			delete temp1;
		}
	        else                                                //左右子树中至少有一个不存在的情况
		{
			temp1 = b;
			if (b->rchild != NULL)
			{
				temp1 = b->rchild;
				b->info = temp1->info;
				b->rchild = temp1->rchild;
				b->lchild = temp1->lchild;
			}
			else if (b->lchild != NULL)
			{
				temp1 = b->lchild;
				b->info = temp1->info;
				b->rchild = temp1->rchild;
				b->lchild = temp1->lchild;
			}
			else if (b == root) root = NULL;
			else if (a->rchild == temp1) a->rchild = NULL;
			else a->lchild = NULL;
			delete temp1;
		}
	
}
      以上代码皆经测试通过,欢迎交流。





  • 3
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值