在计算机科学中,二叉树是每个结点最多有两个子树的树结构。通常子树被称作“左子树”(left subtree)和“右子树”(right subtree)。二叉树常被用于实现二叉查找树和二叉堆。
二叉树是递归定义的,其结点有左右子树之分,逻辑上二叉树有五种基本形态:
(1)空二叉树——如图(a);
(2)只有一个根结点的二叉树;
(3)只有左子树——如图(c);
(4)只有右子树——如图(d);
(5)完全二叉树——如图(e)。
相关术语:
- 结点 : 树中的⼀个独⽴单元. 包含⼀个数据元素及若⼲指向其他⼦树的分⽀. 例如, A,B,C,D等都是结点;
- 结点的度: 结点拥有的⼦树数称谓结点的度. 例如A的度是3, C的度为1, D的度为3, F的度为0.
- 树的度: 数的度是树内各结点度的最⼤值,例如,上图中的应该是3;
- 叶⼦: 度为0的结点称谓叶⼦或终端结点. 例如,K,J,F,G,M,I,J 都是树的叶⼦.
- ⾮终端结点: 度不为0的结点成为⾮终端结点或分⽀结点. 除了根结点以外,⾮终端结点也称为内部结点;
- 双亲和孩⼦: 结点的⼦树的根称为该结点的孩⼦, 相应地, 该结点称为孩⼦的双亲. 例如,B的双亲为A, B 的孩⼦有E和F.
- 兄弟: 同⼀个双亲的孩⼦之间称为兄弟结点, 例如H,I和J互为兄弟;
- 祖先: 从根到该结点所经历的分⽀上的所有结点, 例如, M的祖先为A,D,H.
- ⼦孙: 以某结点为根的⼦树中的任⼀结点都称为该结点的⼦树. 例如,B的⼦孙为E,F.
- 层次:结点的层次从根开始定义起, 根为第⼀层, 根的孩⼦为第⼆层. 树中任⼀层次等于双亲结点的层次 加1.
- 堂兄弟: 双亲在同⼀层的解答互为堂兄弟. 例如,结点G与E,F,H,i,J 互为堂兄弟.
- 有序树和⽆序树: 如果将树的结点的各⼦树看成从左到右是有次序的(即不能互换)则称为该树为有序树, 否则是⽆序树. 在有序树中最左边的⼦树的根称为第⼀个孩⼦,最右边的称为最后⼀个孩⼦. 什么叫有序 树,就类似在家谱中第⼀房太太,到第五房太太以及孩⼦是有顺序的.这样存在顺序关系叫有序树.
- 节点的⾼度: 节点到叶⼦节点的最⻓路径(边数)
- 节点的深度: 根结点到这个结点所经历的边的个数
- 节点的层数: 节点的深度-1
- 树的⾼度 : 根结点的⾼度.
⼆叉树(Binary Tree) 是n (n>=0)个结点所构成的集合. 它或为空树(n=0),对于⾮空树T:
- 有且仅有⼀个称之为根结点
- 除了根结点以外的其余结点分为2个互不相交的⼦集T1,T2. 分别称为T的左⼦树和右⼦树,且T1和T2本身 都是⼆叉树.
二叉树的特性
- ⼆叉树每个结点⾄多只有2颗⼦树(⼆叉树中不存在度⼤于2的结点). 所以⼆叉树中不存在⼤于2的结点. 注意: 不是只有2个⼦树,⽽是最多只有. 如果⼆叉树中没有⼦树或者只有⼀颗树是可以的.
- ⼆叉树的⼦树有左右之分,其次序不能任意颠倒.类似:就像⼈的双⼿,双脚.有顺序之分
- 即使只有⼀棵树,也需要区分是左⼦树还是右⼦树. 类似: 就像你在⽣活中,摔伤了⼿.伤的是左⼿还是右 ⼿,对你的⽣活影响都是完全不同的
对⼀颗具有n个结点的⼆叉树按层序编号,如果编号为i(1=< i <= n)的结点与同样深度的满⼆叉树中编号为i 的结点⼆叉树中位置完全相同. 则这颗⼆叉树称为完全⼆叉树.
- ⾸先"完全" 和 "满" 的差异, 满⼆叉树⼀定是⼀个完全⼆叉树不⼀定是满的.
- 完全⼆叉树的所有结点和同样深度的满⼆叉树,它们按照层序编号相同的结点⼀⼀对应. 这⾥有⼀个关键 词是按层序编号.
- 叶⼦结点只能出现在最下两层
- 最下层的叶⼦⼀定集中在左部连接
- 倒数第⼆层,若有叶⼦节点, ⼀定都在右部连续位置
- 如果结点度为1, 则该结点只有左孩⼦, 既不存在只有右⼦树的情况
- 同样结点数的⼆叉树, 完全⼆叉树的深度最⼩;
了解了二叉树的一些基本概念后,接下来通过顺序存储和链式存储来实现二叉树的一些基本操作
二叉树的顺序存储
二叉树的顺序存储可以直接利用数组来进行存储,通过数组来实现一系列的操作
用数组实现二叉树清空,直接利用跟二叉树的初始化一样,但是为了区分开来,我们用宏定义来换了一个名字
//构建空二叉树
Status InitBiTree(SqBiTree T){
for (int i = 0; i<MAX_TREE_SIZE; i++) {
T[i] = Nil;
}
return OK;
}
//按层次序依次输入二叉树节点值
Status CreateBiTree(SqBiTree T){
int i = 0;
while (i<10) {
T[i] = i+1;
//节点不为空,且有父节点
if(i!=0 && T[(i+1)/2-1] == Nil && T[i]!=Nil){
printf("出现无父节点的非根节点%d\n",T[i]);
exit(ERROR);
}
i++;
}
//将空值赋值给后面的节点
while (i<MAX_TREE_SIZE) {
T[i] = Nil;
i++;
}
return OK;
}
//清除二叉树,根初始化二叉树一摸一样
#define ClearBiTree InitBiTree
//判断二叉树是否为空
Status BiTreeEmpty(SqBiTree T){
if(T[0] == Nil){
return TRUE;
}
return FALSE;
}
根据二叉树的特性,第0层时,数量为1,第1层时,数量为2,第三层数量为8……第n层时,数量为2^n-1
/*
返回二叉树的深度
初始条件:二叉树是否存在,e是T中某个节点的位置
操作结构:返回处于位置e(层,本层序号)的节点值
*/
int BiTreeDepth(SqBiTree T){
int j = -1;
int i;
//找到最后一个节点
for (i = MAX_TREE_SIZE-1; i>0; i--) {
if(T[i]!=Nil)
break;
}
do {
j++;
} while (powl(2, j)<=i);
return j;
}
定位二叉树的节点的位置,查找根节点,对固定节点的值进行替换
/*返回处于位置e(层,本层序号)的结点值
初始条件: 二叉树T存在,e是T中某个结点(的位置)
操作结构: 返回处于位置e(层,本层序号)的结点值
*/
ElemType Value(SqBiTree T,Position e){
/*
Position.level -> 结点层.表示第几层;
Position.order -> 本层的序号(按照满二叉树给定序号规则)
*/
//pow(2,e.level-1) 找到层序
printf("%d\n",(int)pow(2, e.level-1));
//e.order
printf("%d\n",e.order);
return T[(int)pow(2, e.level-1)+e.order-2];
}
/* 获取二叉树跟结点的值
初始条件: 二叉树T存在
操作结果: 当T不空,用e返回T的根, 返回OK; 否则返回ERROR
*/
Status Root(SqBiTree T,ElemType *e){
if(BiTreeEmpty(T)){
return ERROR;
}
*e = T[0];
return OK;
}
/*
给处于位置e的结点赋值
初始条件: 二叉树存在,e是T中某个结点的位置
操作结果: 给处于位置e的结点赋值Value;
*/
Status Assign(SqBiTree T,Position p,ElemType e){
//找到当前e的位置
int i = (int)powl(2, p.level-1)+p.order-2;
//叶子节点的双亲
if(e!=Nil && T[(i+1)/2-1] == Nil){
return ERROR;
}
//给双亲节点赋值
if(e == Nil && (T[i*2+1]!=Nil || T[i*2+2]!=Nil)){
return ERROR;
}
T[i] = e;
return OK;
}
找到一个节点的父节点,兄弟节点,都需要找到当前节点的值,在根据值的顺序求出另一个节点
/*
获取e的双亲;
初始条件: 二叉树存在,e是T中的某一个结点
操作结果: 若e是T的非根结点, 则返回它的双亲,否则返回"空"
*/
ElemType Parent(SqBiTree T,ElemType e){
//判空
if(T[0] == Nil){
return ERROR;
}
for (int i = 1; i<MAX_TREE_SIZE; i++) {
//找到e
if(T[i] == e){
return T[(i+1)/2-1];
}
}
//没有找到
return Nil;
}
/*
获取某个结点的左孩子;
初始条件:二叉树T存在,e是某个结点
操作结果:返回e的左孩子,若e无左孩子,则返回"空"
*/
ElemType LeftChild(SqBiTree T,ElemType e){
//判空
if(T[0] == Nil){
return ERROR;
}
for (int i = 0; i<MAX_TREE_SIZE-1; i++) {
//找到e
if(T[i] == e){
return T[i*2+1];
}
}
return Nil;
}
/*
获取某个结点的右孩子;
初始条件:二叉树T存在,e是某个结点
操作结果:返回e的左孩子,若e无左孩子,则返回"空"
*/
ElemType RightChild(SqBiTree T,ElemType e){
//判空
if(T[0] == Nil){
return ERROR;
}
for (int i = 0; i<MAX_TREE_SIZE-1; i++) {
//找到e
if(T[i] == e){
return T[i*2+2];
}
}
return Nil;
}
/*
获取结点的左兄弟
初始条件: 二叉树T存在,e是T中某个结点
操作结果: 返回e的左兄弟。若e是T的左孩子或无左兄弟,则返回"空"
*/
ElemType LeftSibling(SqBiTree T,ElemType e){
//判空
if(T[0] == Nil){
return ERROR;
}
for (int i = 1; i<=MAX_TREE_SIZE; i++) {
if(T[i] == e&&i%2 == 0){
return T[i-1];
}
}
return Nil;
}
/* 获取结点的右兄弟
初始条件: 二叉树T存在,e是T中某个结点
操作结果: 返回e的右兄弟。若e是T的右孩子或无右兄弟,则返回"空"
*/
ElemType RightSibling(SqBiTree T,ElemType e){
//判空
if(T[0] == Nil){
return ERROR;
}
for (int i = 1; i<=MAX_TREE_SIZE; i++) {
if(T[i] == e&&i%2 == 1){
return T[i+1];
}
}
return Nil;
}
对二叉树进行前序,中序,后序和层序遍历
//1、层序遍历
void LevelOrderTree(SqBiTree T){
int i = MAX_TREE_SIZE-1;
//找到最后一个非空节点序号
while (T[i] == Nil) {
i--;
}
//从根节点开始,遍历
for (int j = 0; j<=i; j++) {
if(T[j] != Nil)
printf("%3d",T[j]);
}
printf("\n");
}
void preDisplay(SqBiTree T,int e){
if(!BiTreeEmpty(T)){
//打印根节点
printf("%3d",T[e]);
//先序遍历左子树
if(T[2*e+1]!=Nil)
preDisplay(T, 2*e+1);
//后序遍历右子树
if(T[2*e+2]!=Nil)
preDisplay(T, 2*e+2);
}
// printf("\n");
}
void InDisplay(SqBiTree T,int e){
if(!BiTreeEmpty(T)){
//先序遍历左子树
if(T[2*e+1]!=Nil)
InDisplay(T, 2*e+1);
printf("%3d",T[e]);
//后序遍历右子树
if(T[2*e+2]!=Nil)
InDisplay(T, 2*e+2);
}
// printf("\n");
}
void backDisplay(SqBiTree T,int e){
if(!BiTreeEmpty(T)){
//先序遍历左子树
if(T[2*e+1]!=Nil)
backDisplay(T, 2*e+1);
//后序遍历右子树
if(T[2*e+2]!=Nil)
backDisplay(T, 2*e+2);
printf("%3d",T[e]);
}
// printf("\n");
}
main函数调用
Status iStatus;
Position p;
ElemType e;
SqBiTree T;
InitBiTree(T);
CreateBiTree(T);
printf("建立二叉树后,树空否?%d(1:是 0:否)\n",BiTreeEmpty(T));
printf("树的深度=%d\n",BiTreeDepth(T));
p.level = 3;
p.order = 2;
e = Value(T, p);
printf("第%d层第%d个节点的值:%d\n",p.level,p.order,e);
iStatus = Root(T, &e);
if(iStatus){
printf("二叉树的根为:%d\n",e);
}else{
printf("树为空,无根!\n");
}
e = 99;
Assign(T, p, e);
e = Value(T, p);
printf("第%d层第%d个节点的值:%d\n",p.level,p.order,e);
printf("节点%d的父节点为:%d\n",e,Parent(T, e));
printf("左右兄弟节点分别为:%d,%d\n",LeftChild(T, e),RightChild(T, e));
printf("节点%d的左右兄弟节点:%d,%d\n",e,LeftSibling(T, e),RightSibling(T, e));
Assign(T, p, 5);
printf("二叉树T层序遍历:");
LevelOrderTree(T);
printf("\n");
printf("二叉树T先序遍历:");
preDisplay(T,0);
printf("\n");
printf("二叉树T中序遍历:");
InDisplay(T,0);
printf("\n");
printf("二叉树T后序遍历:");
backDisplay(T, 0);
printf("\n");

二叉树的链式存储
构建二叉树
typedef char ElemType;
ElemType Nil = ' ';
typedef struct BitNode{
ElemType data;
struct BitNode *lChild,*rChild;
}BiTNode,*BiTree;
//构建二叉树
Status InitBiTree(BiTree *T){
*T = NULL;
return OK;
}
二叉树的销毁和清空,性质都一样,需要释放节点
/* 销毁二叉树
初始条件: 二叉树T存在。
操作结果: 销毁二叉树T
*/
void DestoryBiTree(BiTree *T){
if(*T){
if((*T)->lChild)
DestoryBiTree(&(*T)->lChild);
if((*T)->rChild)
DestoryBiTree(&(*T)->rChild);
free(*T);
*T = NULL;
}
}
#define ClearBiTree DestoryBiTree
void CreateBiTree(BiTree *T){
ElemType ch;
//获取字符
ch = str[indexs++];
//判断当前字符是不是'#'
if(ch == '#'){
*T = NULL;
}else{
//创建新的节点
*T = (BiTree)malloc(sizeof(BiTNode));
//是否创建成功
if (!*T) {
exit(OVERFLOW);
}
/* 生成根结点 */
(*T)->data = ch;
/* 构造左子树 */
CreateBiTree(&(*T)->lChild);
/* 构造右子树 */
CreateBiTree(&(*T)->rChild);
}
}
判断二叉树是否为空,直接判断根节点的值是否为空
对二叉树的深度求值,需要对左子树和右子树进行递归层层遍历,得到的两个值进行比较
/*
二叉树T是否为空;
初始条件: 二叉树T存在
操作结果: 若T为空二叉树,则返回TRUE,否则FALSE
*/
Status BiTreeEmpty(BiTree T){
if(T)
return FALSE;
else
return TRUE;
}
/*
二叉树T的深度
初始条件: 二叉树T存在
操作结果: 返回T的深度
*/
int BiTreeDepth(BiTree T){
int i,j;
if(!T)
return 0;
//计算左孩子的深度
if(T->lChild)
i=BiTreeDepth(T->lChild);
else
i=0;
//计算右孩子的深度
if(T->rChild)
j=BiTreeDepth(T->rChild);
else
j=0;
//比较i和j
return i>j?i+1:j+1;
}
/*
二叉树T的根
初始条件: 二叉树T存在
操作结果: 返回T的根
*/
ElemType Root(BiTree T){
if(BiTreeEmpty(T))
return Nil;
return T->data;
}
前序、中序、后序遍历
#pragma mark--二叉树遍历
/*
前序递归遍历T
初始条件:二叉树T存在;
操作结果: 前序递归遍历T
*/
void preDisplay(BiTree T){
if(T == NULL)
return;
printf("%2c",T->data);
preDisplay(T->lChild);
preDisplay(T->rChild);
}
/*
中序递归遍历T
初始条件:二叉树T存在;
操作结果: 中序递归遍历T
*/
void InDisplay(BiTree T){
if(T == NULL)
return;
InDisplay(T->lChild);
printf("%2c",T->data);
InDisplay(T->rChild);
}
/*
后序递归遍历T
初始条件:二叉树T存在;
操作结果: 中序递归遍历T
*/
void backDisplay(BiTree T){
if(T == NULL)
return;
backDisplay(T->lChild);
backDisplay(T->rChild);
printf("%2c",T->data);
}
main函数调用
BiTree T;
ElemType e;
InitBiTree(&T);
StrAssign(str, "ABDH#K###E##CFI###G#J##");
CreateBiTree(&T);
printf("判断二叉树是否为空:%d(1:是 0:否),树的深度=%d\n",BiTreeEmpty(T),BiTreeDepth(T));
e = Root(T);
printf("二叉树的根为:%c\n",e);
printf("二叉树的前序遍历:");
preDisplay(T);
printf("\n");
printf("二叉树的中序遍历:");
InDisplay(T);
printf("\n");
printf("二叉树的后序遍历:");
backDisplay(T);
printf("\n");
打印结果

网友评论