关于二叉树的遍历和建立总结

一 树

1.树是包含N(node)个节点的有穷集。其中:

  • 1)每个元素称为结点。
    2)每个结点有零个或多个子结点,单个结点是一棵树,树根就是该结点本身。空集合也是树,称为空树。空树中没有结点。
    3)
    2.结点是数据结构中的基础,是构成复杂数据结构的基本组成单位。
    结点拥有的子树数目称为结点的度。
    结点子树的根结点为该结点的孩子结点。相应该结点称为孩子结点的双亲结点。
    图中,A为B的双亲结点,B为A的孩子结点。
    同一个双亲结点的孩子结点之间互称兄弟结点。
    图中,结点B与结点C互为兄弟结点。

在这里插入图片描述

3.从根开始定义起,根为第一层,根的孩子为第二层,以此类推。
树中结点的最大层次数称为树的深度或高度。如图,所示树的深度为4。
在这里插入图片描述

二 二叉树

1.二叉树是n(n>=0)个结点的有限集合,该集合或者为空集(称为空二叉树),或者由一个根结点和两棵互不相交的、分别称为根结点的左子树和右子树组成。
由二叉树定义以及图示分析得出二叉树有以下特点:
1)每个结点最多有两颗子树,所以二叉树中不存在度大于2的结点。
2)左子树和右子树是有顺序的,次序不能任意颠倒。
3)即使树中某结点只有一棵子树,也要区分它是左子树还是右子树。

  • 二叉树性质:
  • 1)在二叉树的第i层上最多有2i-1 个节点 。(i>=1)
    2)二叉树中如果深度为k,那么最多有2k-1个节点。(k>=1) 3)n0=n2+1 n0表示度数为0的节点数,n2表示度数为2的节点数。
    4)在完全二叉树中,具有n个节点的完全二叉树的深度为[log2n]+1,其中[log2n]是向下取整。
    5)若对含 n个结点的完全二叉树从上到下且从左至右进行 1 至 n 的编号,则对完全二叉树中任意一个编号为 i 的结点有如下特性:
  • (1)若 i=1,则该结点是二叉树的根,无双亲, 否则,编号为 [i/2] 的结点为其双亲结点;
  • (2)若 2i>n,则该结点无左孩子, 否则,编号为 2i 的结点为其左孩子结点;
    (3) 若 2i+1>n,则该结点无右孩子结点, 否则,编号为2i+1 的结点为其右孩子结点

2.满二叉树:在一棵二叉树中。如果所有分支结点都存在左子树和右子树,并且所有叶子都在同一层上,这样的二叉树称为满二叉树。
满二叉树的特点有:
1)叶子只能出现在最下一层。出现在其它层就不可能达成平衡。
2)非叶子结点的度一定是2。
3)在同样深度的二叉树中,满二叉树的结点个数最多,叶子数最多。
在这里插入图片描述
3**.完全二叉树:设二叉树的深度为h,除第 h 层外,其它各层 (1~h-1) 的结点数都达到最大个数,第 h 层所有的结点都连续集中在最左边,这就是完全二叉树。
特点:

  • 1)叶子结点只能出现在最下层和次下层。
  • 2)最下层的叶子结点集中在树的左部。
  • 3)倒数第二层若存在叶子结点,一定在右部连续位置。
    4)如果结点度为1,则该结点只有左孩子,即没有右子树。
    5)同样结点数目的二叉树,完全二叉树深度最小。
    注:满二叉树一定是完全二叉树,但反过来不一定成立
    在这里插入图片描述

三 二叉树的存储结构

1.二叉树的顺序存储结构就是使用一维数组存储二叉树中的结点,并且结点的存储位置,就是数组的下标索引。
在这里插入图片描述
在这里插入图片描述
由图3.7可以看出,当二叉树为完全二叉树时,结点数刚好填满数组。
那么当二叉树不为完全二叉树时,采用顺序存储形式如何呢?例如:对于图3.8描述的二叉树:
在这里插入图片描述
在这里插入图片描述
2.既然顺序存储不能满足二叉树的存储需求,那么考虑采用链式存储。由二叉树定义可知,二叉树的每个结点最多有两个孩子。因此,可以将结点数据结构定义为一个数据和两个指针域。表示方式如图3.11所示:
在这里插入图片描述
定义结点代码:

//二叉树的链式存储
/定义节点的结构
typedef char TElemType;
typedef struct BiTNode
{
	TElemType data;//定义数据
	BiTNode *rchild,lchild;//定义左右孩子
}*BiTree,BiTNode;



///
//二叉树的顺序存储
/定义节点的结构,本质上是一个结构体数组
typedef char datatpye;
typedef struct node
{
	datatype data;
	int lchild,rchild;
	int parent;
}Node;

3.二叉树遍历
在这里插入图片描述
1)在二叉树中最重要的操作大概就是遍历,如链表这样的数据结构,遍历的方式是唯一的,因为我们只知道链表的头结点,遍历到一个结点时也只知道下一个结点(单链表),但是在树中却有多种遍历方式,通常有:
前序遍历:根结点—>左子结点—>右子结点,10、6、4、8、14、12、16;
中序遍历:左子结点—>根结点—>右子结点,4、6、8、10、12、14、16;
后序遍历:左子结点—>右子结点—>根结点,4、8、6、12、16、14、10;
层序遍历:第一层—>第二层—>第n层,10、6、14、4、8、12、16;

2)虽然二叉树的遍历过程看似繁琐,但是由于二叉树是一种递归定义的结构,故采用递归方式遍历二叉树的代码十分简单.

/二叉树遍历
///
//二叉树前序遍历递归方式,根-左-右
void BiT_preorder(BiTree T)
{
	if(T==null)//判断二叉树是否为空
		return;//返回空
	printf("%c",T->data)//显示节点数据,根节点。
	BiT_preorder(T->lchild);//然后递归遍历左子树
	BiT_perorder(T->rchild);//然后递归遍历右子树
}
//
//二叉树中序遍历递归方式,左-根-右
void BiT_midorder(BiTree T)
{
	if(T==null)//判断二叉树是否为空
		return;//返回空
	BiT_midorder(T->lchild);//递归遍历左子树
	printf("%c",T->data);//显示节点数据,根节点。
	BiT_midorder(T->rchild);//递归遍历右子树
}
///
//二叉树后序遍历递归方式,左-右-根
void BiT_postorder(BiTree T)
{
	if(T==null)//判断二叉树是否为空
		return;//返回空
	BiT_postorder(T->lchild);//递归遍历左子树
	BiT_postorder(T->rchild);//递归遍历右子树
	printf("%c",T->data);//显示节点数据,根节点。
}

//层次遍历:
//实现过程 
//1、创建一个指针数组,保存二叉树结构体指针, 
//2、保存二叉树根节点,再申请变量 in、out ,控制数组,在遍历过程中,
//始终能找到节点和该节点的前一个节点, 
//3、循环以上过程。
///层次遍历:队列实现一层一层。
void BiT_floorder(BiTree T)
{
	BiTree tmp[N];//创建pTreeNode指针类型的指针数组
	int in=0;
	int out=0;
	tmp[in++]=T;//先保存二叉树根节点 
	while(in>out)
	{
		if(tmp[out])
		{
			printf("%c",tmp[out]->data);
			tmp[in++]=tmp[out]->lchild;
			tmp[in++]=tmp[out]->rchild;
		}
		out++;
	}
}

最后完整代码:

#include<stdio.h>
#include<stdlib.h>
#include<iostream.h>
#include<assert.h>
#define N 100
/
//二叉树的链式存储
/定义节点的结构
typedef char TElemType;
typedef struct BiTNode
{
	TElemType data;//定义数据
	BiTNode *rchild,*lchild;//定义左右孩子
}*BiTree,BiTNode;


//二叉树的顺序存储
/定义节点的结构,本质上是一个结构体数组
/*/typedef char datatype;
typedef struct node
{
	datatype data;
	int lchild,rchild;
	int parent;
}Node;*/
void InitBiTree(BiTree *T);
void create_BiTree(BiTree *T);
void BiT_preorder(BiTree T);
void BiT_midorder(BiTree T);
void BiT_postorder(BiTree T);
void BiT_floorder(BiTree T);
//
//初始化二叉树,建立一个空树
void InitBiTree(BiTree *T)
{
	*T=NULL;
	
}

///
//二叉树的建立,前序遍历递归方式,根-左-右建立。
void create_BiTree(BiTree *T)
{
	TElemType ch;
	scanf("%c",&ch);
	if(ch=='#')//#表示为空,判断是否为空节点。
	{T=NULL;}
	else
	{
		*T=(BiTree)malloc(sizeof(BiTNode));//创造二叉树内存空间
		if(!*T)
		{ exit(0);}
		else
		{
			(*T)->data=ch;//建立根节点
			(*T)->lchild=NULL;
			(*T)->rchild=NULL;
			create_BiTree(&(*T)->lchild);//构造左
			create_BiTree(&(*T)->rchild);//构造右
		}
	}
}

/二叉树遍历
///
//二叉树前序遍历递归方式,根-左-右
void BiT_preorder(BiTree T)
{
	if(T==NULL)//判断二叉树是否为空
		return;//返回空
	printf("%c",T->data);//显示节点数据,根节点。
	BiT_preorder(T->lchild);//然后递归遍历左子树
	BiT_preorder(T->rchild);//然后递归遍历右子树
}
//
//二叉树中序遍历递归方式,左-根-右
void BiT_midorder(BiTree T)
{
	if(T==NULL)//判断二叉树是否为空
		return;//返回空
	BiT_midorder(T->lchild);//递归遍历左子树
	printf("%c",T->data);//显示节点数据,根节点。
	BiT_midorder(T->rchild);//递归遍历右子树
}
///
//二叉树后序遍历递归方式,左-右-根
void BiT_postorder(BiTree T)
{
	if(T==NULL)//判断二叉树是否为空
		return;//返回空
	BiT_postorder(T->lchild);//递归遍历左子树
	BiT_postorder(T->rchild);//递归遍历右子树
	printf("%c",T->data);//显示节点数据,根节点。
}
///层次遍历:队列实现一层一层。
void BiT_floorder(BiTree T)
{
	BiTree tmp[N];//创建pTreeNode指针类型的指针数组
	int in=0;
	int out=0;
	tmp[in++]=T;//先保存二叉树根节点 
	while(in>out)
	{
		if(tmp[out])
		{
			printf("%c",tmp[out]->data);
			tmp[in++]=tmp[out]->lchild;
			tmp[in++]=tmp[out]->rchild;
		}
		out++;
	}
}

void main()
{
		BiTree T;
	InitBiTree(&T);
	create_BiTree(&T);

	printf("前序遍历:");
	BiT_preorder(T);

    printf("\n");
	printf("中序遍历:");
	BiT_midorder(T);

	
    printf("\n");
	printf("后序遍历:");
	BiT_postorder(T);

   
    printf("\n");
	printf("层次遍历:");
	BiT_floorder(T);
	 printf("\n");
}

//注意:在输入是要按照二叉树的遍历输入:如
// A
//B C,要输入为AB##C##.需要输入叶子节点或单分支节点的空指针用’#'代替
在这里插入图片描述
参考:
[1]: https://blog.csdn.net/chaipp0607/article/details/77050948
[2]: https://www.jianshu.com/p/bf73c8d50dc2

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值