将二叉排序树转换成排序双向链表,并返回最左边的结点(不建新结点)

二叉排序树是带根结点的二叉树,其内部结点各自存储一个关键字(以及可选的相关值),并且各自具有两个可区分的子树,通常表示为左和右。该树还拥有二分搜索的属性,该属性声明每个结点中的关键字必须大于或等于左子树中存储的任何关键字,并且小于或等于右子树中存储的任何关键字.

二叉树转换成双向链表,中序遍历正好是值从小到大,只需用一个p指针保存中序遍历的前一个结点。因为是中序遍历,遍历顺序就是双线链表的建立顺序;每一个结点访问时它的左子树肯定被访问过了,所以放心大胆的改它的left指针,不怕树断掉;同理,p指向的结点保存的数肯定小于当前结点,所以其左右子树肯定都访问过了,所以其right指针也可以直接改。最后需要一直向左找到双向链表的头结点。

#include <stdio.h>
#include <stdlib.h>
#define n 8
#define MAXSIZE 50
typedef int DataType;
typedef struct BiNode{
	DataType data;
	struct BiNode *lchild,*rchild;
}BiNode,*BiTree;
BiTree p=NULL;
BiTree SearchildNode(BiTree *T,DataType x){
	BiTree p,q;
	p=q=*T;
	while(q){
		if(x>q->data){
			p=q;
			q=q->rchild;
		}else if(x<q->data){
			p=q;
			q=q->lchild;
		}else break;
	}
	return q==NULL?p:q;//没有,return爹(便于InsertNode);有,return该结点
}
void InsertNode(BiTree *T,DataType x){
	BiTree p,s;
	s=(BiTree)malloc(sizeof(BiNode));
	s->data=x;
	s->lchild=s->rchild=NULL;
	p=SearchildNode(T,x);
	if(!p)*T=s;//第一把
	else{
		if(p->data==x){free(s);}//已经有了,则不插
		else{
			if(x>p->data)p->rchild=s;
			else p->lchild=s;
		}
	}
}
void InOrderTraverse(BiTree T){
	if(T==NULL)return;
	else{
		InOrderTraverse(T->lchild);
		T->lchild=p;
		if(p)p->rchild=T;
		p=T;
		InOrderTraverse(T->rchild);
	}
}
BiTree convertTreeToList(BiTree T){
	InOrderTraverse(T);
	while(T&&T->lchild)T=T->lchild;
	return T;
}
void main(){
	int i,a[n]={18,3,7,12,5,9,10,1};
	BiTree T=NULL,temp;
	for(i=0;i<n;i++){InsertNode(&T,a[i]);}
	temp=convertTreeToList(T);
	while(temp){printf("%d ",temp->data);temp=temp->rchild;}
	system("pause");
}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值