C语言|数据结构——树的定义、存储与遍历

简介: 基本概念定义:1.有且只有一个称为根的节点; 2.有若干个互不相交的子树,这些子树本身也是一棵树; 3.由节点和边组组成的; 4.每个节点只有一个父节点,可以有无数个子节点(除了根节点)。分类:|一般树。任意一个子节点个数不受限制,可以是有序树也可以是无序树。|二叉树。任意一个节点最大度为2,二叉树是有序树,左右节点不能随意互换。 | 一般二叉树 |满二叉树。每一层节点都是满的。 |完全二叉树。除最后一层外,每一层节点都是满的,最后一层节点一定从左向右连续排列。|森林。n个互不相交的树的集合,可以是互不相连的几个树一些专业术语:父节

基本概念

定义:

1.有且只有一个称为根的节点;  

2.有若干个互不相交的子树,这些子树本身也是一棵树;  

3.由节点和边组组成的;  

4.每个节点只有一个父节点,可以有无数个子节点(除了根节点)。

分类:

|一般树。任意一个子节点个数不受限制,可以是有序树也可以是无序树。

|二叉树。任意一个节点最大度为2,二叉树是有序树,左右节点不能随意互换。

    | 一般二叉树

    |满二叉树。每一层节点都是满的。

    |完全二叉树。除最后一层外,每一层节点都是满的,最后一层节点一定从左向右连续排列。

|森林。n个互不相交的树的集合,可以是互不相连的几个树

一些专业术语:

父节点:该节点上面紧挨着的一个节点,一个节点只有一个父节点。

祖先节点:一个节点的父节点的父节点,一个节点只有一个。

兄弟节点:两个节点具有共同的父节点。

堂兄弟节点:两个节点的父节点不同,但各自的父节点彼此是兄弟节点。

子孙节点:一个节点的所有子节点和子孙节点。

树的深度:从根节点到最底层节点的层数,根节点是第一层。

叶子节点:没有子节点的节点。

非终端节点:非叶子节点。

度:子结点的个数称为度。

最大的度:一个树中含有最大度的那个度是这个树最大的度。

树的存储

本质是把非线性结构转化成线性结构进行存储。任何树的存储都是先转化成二叉树之后再按照二叉树的方式进行存储。

二叉树的存储

    |连续存储(只能以完全二叉树形式存储)

    |链式存储

一般树的存储

双亲表示法

*利用结构体数组存储

*数组下表即为该节点编号,结构体内部有两部分,一部分是数据域,一部分存储其父节点编号,如果没有父节点则为-1。

*缺点是只能记录无序树。

孩子表示法

*数组结构体

*每个节点由两部分组成,一部分是数据域,一部分存储其子节点的地址,所有孩子像链表一样连起来。后面的子节点是无序的。

*缺点是难以寻找父节点。

双亲孩子表示法

*数组结构体

*每个节点对应的数组下标就是各自的编号。每个节点由三部分组成:数据域,父节点下标,孩子节点地址。

*优势:即方便找父节点也方便找孩子节点。

二叉树表示法(最常用)

*结构体数组

*思路是先转化为二叉树,再按照二叉树的方法进行存储。每个节点有3个部分组成:一个数据域,一个指向其最左侧第一个孩子,另一个指针域指向其第一个兄弟。这一步可以把一个普通树转化为一个二叉树,这个过程也可以通过画图形象的表示出来。

*普通树转化成二叉树一定没有右子树。

森林的存储

*把森林转化成二叉树。 把根节点互相看做兄弟,接下来转化方法就跟普通树一样。

树的遍历

先序遍历,中序遍历,后序遍历这三种方法有一些共同之处:每遍历到一个节点,这个节点本身就可以看做是又一棵树的根节点。

先序遍历

根节点->先序访问左子树->先序访问右子树

下图遍历顺序:① -> ② -> ④ -> ⑤ -> ③ -> ⑥ -> ⑦

描述:先遍历根节点①;接着来到左孩子②,②这棵树仍然存在左孩子④,所以来到④;④是叶子结点,是一棵空树,左右节点都不存在,所以④这棵树遍历完成,也代表着②的左子树遍历完成;②的左子树遍历完成后,接下来来到其右孩子⑤,⑤的遍历方法和④一样;②的左右节点都遍历完成,②这棵树就遍历完成了,也就是①的左子树遍历完成;接下来以相同的方法遍历①的右子树③,大家自行推断。

        ①


       /    \


    ②     ③


    / \      / \


 ④ ⑤ ⑥ ⑦


中序遍历

左子树->根节点->先序访问右子树

上图遍历顺序:④ -> ② -> ⑤ -> ① -> ⑥ -> ③ -> ⑦

描述:根节点①的左孩子是②,②仍然有左孩子④,直到④才没有左孩子,所以④的左子树可以认为已经遍历完成,然后遍历空树④的根节点④,其右子树也不存在所以也可以认为遍历完成;④遍历完成,即②的左子树遍历完成,接着遍历②的根节点②,接着以同样的操作遍历⑤;②遍历完成,即①的左子树遍历完成,接着以同样的方法遍历其右子树③,请大家自行脑补。

后序遍历

后序遍历左子树->后序遍历右子树->根节点

上图遍历顺序:④ -> ⑤ -> ② -> ⑥ -> ⑦ -> ③ -> ①

描述:下面我只描述一下如何遍历①的右子树。记得我们上面说的吗,“每遍历到一个节点,这个节点本身就可以看做是又一棵树的根节点”,现在我们就把③看作是一棵树的根节点,所以要先遍历其左子树⑥,再遍历其右子树⑦,最后遍历根节点③。

层序遍历

从上往下,从左往右挨个遍历。

上图遍历顺序:① -> ② -> ③ -> ④ -> ⑤ -> ⑥ -> ⑦

#include <stdio.h>#include <stdlib.h> typedefstructT{
intdata;
structT*left;
structT*right;
}T,*PT;
//使用递归的方法创建一个树 PTmake()
{
intdata;
scanf("%d",&data);
getchar();//吸收scanf的回车键 //判断数据,如果是-1,那么不创建该节点并且向上返回//如果不是-1,那么就创建该节点并且把该数字存储到数据域 if(data==-1)
returnNULL;
else    {
PTtree=(PT)malloc(sizeof(T));
tree->data=data;
printf("请输入节点%d的左子树",data);
tree->left=make();
printf("请输入节点%d的右子树",data);
tree->right=make();
returntree;
    }
}
//先序遍历 :根-左-右 voidxianxu(PTtree)
{
//判断树是否为空 if(tree==NULL)
return;
else    {
printf("%d",tree->data); 
xianxu(tree->left);
xianxu(tree->right); 
    }
} 
//中序遍历:左-根-右 voidzhongxu(PTtree)
{
if(tree==NULL)
return;
else    {
xianxu(tree->left);
printf("%d",tree->data); 
xianxu(tree->right); 
    }
}
//后序遍历:左-右-根 voidhouxu(PTtree)
{
if(tree==NULL)
return;
else    {
xianxu(tree->left);
xianxu(tree->right); 
printf("%d",tree->data); 
    }
}
intmain()
{
printf("创建一棵树,请输入根节点");
PTtree=make();
intchoose;
printf("先序遍历请按1;中序遍历请按2;后序遍历请按3:\n");
scanf("%d",&choose);
getchar();
switch(choose)
    {
case1:xianxu(tree);
break;
case2:zhongxu(tree);
break;
case3:houxu(tree);
break;
default:
break;
    }
 } 


相关文章
|
8天前
|
C语言
【数据结构】栈和队列(c语言实现)(附源码)
本文介绍了栈和队列两种数据结构。栈是一种只能在一端进行插入和删除操作的线性表,遵循“先进后出”原则;队列则在一端插入、另一端删除,遵循“先进先出”原则。文章详细讲解了栈和队列的结构定义、方法声明及实现,并提供了完整的代码示例。栈和队列在实际应用中非常广泛,如二叉树的层序遍历和快速排序的非递归实现等。
75 9
|
7天前
|
存储 搜索推荐 算法
【数据结构】树型结构详解 + 堆的实现(c语言)(附源码)
本文介绍了树和二叉树的基本概念及结构,重点讲解了堆这一重要的数据结构。堆是一种特殊的完全二叉树,常用于实现优先队列和高效的排序算法(如堆排序)。文章详细描述了堆的性质、存储方式及其实现方法,包括插入、删除和取堆顶数据等操作的具体实现。通过这些内容,读者可以全面了解堆的原理和应用。
48 16
|
7天前
|
C语言
【数据结构】二叉树(c语言)(附源码)
本文介绍了如何使用链式结构实现二叉树的基本功能,包括前序、中序、后序和层序遍历,统计节点个数和树的高度,查找节点,判断是否为完全二叉树,以及销毁二叉树。通过手动创建一棵二叉树,详细讲解了每个功能的实现方法和代码示例,帮助读者深入理解递归和数据结构的应用。
41 8
|
10天前
|
存储 C语言
【数据结构】手把手教你单链表(c语言)(附源码)
本文介绍了单链表的基本概念、结构定义及其实现方法。单链表是一种内存地址不连续但逻辑顺序连续的数据结构,每个节点包含数据域和指针域。文章详细讲解了单链表的常见操作,如头插、尾插、头删、尾删、查找、指定位置插入和删除等,并提供了完整的C语言代码示例。通过学习单链表,可以更好地理解数据结构的底层逻辑,提高编程能力。
37 4
|
10天前
|
C语言
【数据结构】双向带头循环链表(c语言)(附源码)
本文介绍了双向带头循环链表的概念和实现。双向带头循环链表具有三个关键点:双向、带头和循环。与单链表相比,它的头插、尾插、头删、尾删等操作的时间复杂度均为O(1),提高了运行效率。文章详细讲解了链表的结构定义、方法声明和实现,包括创建新节点、初始化、打印、判断是否为空、插入和删除节点等操作。最后提供了完整的代码示例。
28 0
|
29天前
|
C语言 C++
C语言 之 内存函数
C语言 之 内存函数
32 3
|
C语言 存储
软件设计师2006年11月下午试题5(C语言 树及其孩子-兄弟表示)
[说明]  一般的树结构常采用孩子-兄弟表示法表示,即用二叉链表作树的存储结构,链表中结点的两个链域分别指向该结点的第一个孩子结点和下一个兄弟结点。例如,图 5-1(a) 所示的树的孩子-兄弟表示如图 5-1(b)所示。
948 0
|
20天前
|
存储 缓存 C语言
【c语言】简单的算术操作符、输入输出函数
本文介绍了C语言中的算术操作符、赋值操作符、单目操作符以及输入输出函数 `printf` 和 `scanf` 的基本用法。算术操作符包括加、减、乘、除和求余,其中除法和求余运算有特殊规则。赋值操作符用于给变量赋值,并支持复合赋值。单目操作符包括自增自减、正负号和强制类型转换。输入输出函数 `printf` 和 `scanf` 用于格式化输入和输出,支持多种占位符和格式控制。通过示例代码详细解释了这些操作符和函数的使用方法。
33 10
|
13天前
|
存储 算法 程序员
C语言:库函数
C语言的库函数是预定义的函数,用于执行常见的编程任务,如输入输出、字符串处理、数学运算等。使用库函数可以简化编程工作,提高开发效率。C标准库提供了丰富的函数,满足各种需求。
|
19天前
|
机器学习/深度学习 C语言
【c语言】一篇文章搞懂函数递归
本文详细介绍了函数递归的概念、思想及其限制条件,并通过求阶乘、打印整数每一位和求斐波那契数等实例,展示了递归的应用。递归的核心在于将大问题分解为小问题,但需注意递归可能导致效率低下和栈溢出的问题。文章最后总结了递归的优缺点,提醒读者在实际编程中合理使用递归。
47 7
下一篇
无影云桌面