二叉排序树是带根结点的二叉树,其内部结点各自存储一个关键字(以及可选的相关值),并且各自具有两个可区分的子树,通常表示为左和右。该树还拥有二分搜索的属性,该属性声明每个结点中的关键字必须大于或等于左子树中存储的任何关键字,并且小于或等于右子树中存储的任何关键字.
二叉树转换成双向链表,中序遍历正好是值从小到大,只需用一个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");
}