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;
    }
 } 


相关文章
|
20天前
|
算法 数据处理 C语言
C语言中的位运算技巧,涵盖基本概念、应用场景、实用技巧及示例代码,并讨论了位运算的性能优势及其与其他数据结构和算法的结合
本文深入解析了C语言中的位运算技巧,涵盖基本概念、应用场景、实用技巧及示例代码,并讨论了位运算的性能优势及其与其他数据结构和算法的结合,旨在帮助读者掌握这一高效的数据处理方法。
30 1
|
28天前
|
存储 算法 搜索推荐
【趣学C语言和数据结构100例】91-95
本文涵盖多个经典算法问题的C语言实现,包括堆排序、归并排序、从长整型变量中提取偶数位数、工人信息排序及无向图是否为树的判断。通过这些问题,读者可以深入了解排序算法、数据处理方法和图论基础知识,提升编程能力和算法理解。
44 4
|
28天前
|
存储 机器学习/深度学习 搜索推荐
【趣学C语言和数据结构100例】86-90
本文介绍并用C语言实现了五种经典排序算法:直接插入排序、折半插入排序、冒泡排序、快速排序和简单选择排序。每种算法都有其特点和适用场景,如直接插入排序适合小规模或基本有序的数据,快速排序则适用于大规模数据集,具有较高的效率。通过学习这些算法,读者可以加深对数据结构和算法设计的理解,提升解决实际问题的能力。
43 4
|
28天前
|
存储 算法 数据处理
【趣学C语言和数据结构100例】81-85
本文介绍了五个经典算法问题及其C语言实现,涵盖图论与树结构的基础知识。包括使用BFS求解单源最短路径、统计有向图中入度或出度为0的点数、统计无向无权图各顶点的度、折半查找及二叉排序树的查找。这些算法不仅理论意义重大,且在实际应用中极为广泛,有助于提升编程能力和数据结构理解。
38 4
|
28天前
|
算法 数据可视化 数据建模
【趣学C语言和数据结构100例】76-80
本文介绍了五种图论算法的C语言实现,涵盖二叉树的层次遍历及广度优先搜索(BFS)和深度优先搜索(DFS)的邻接表与邻接矩阵实现。层次遍历使用队列按层访问二叉树节点;BFS利用队列从源节点逐层遍历图节点,适用于最短路径等问题;DFS通过递归或栈深入图的分支,适合拓扑排序等场景。这些算法是数据结构和算法学习的基础,对提升编程能力和解决实际问题至关重要。
45 4
|
28天前
|
存储 算法 vr&ar
【趣学C语言和数据结构100例】71-75
本文介绍了五个C语言数据结构问题及其实现,涵盖链表与二叉树操作,包括按奇偶分解链表、交换二叉树左右子树、查找节点的双亲节点、计算二叉树深度及求最大关键值。通过递归和遍历等方法,解决了理论与实际应用中的常见问题,有助于提升编程能力和数据结构理解。
36 4
|
28天前
|
存储 算法 C语言
【趣学C语言和数据结构100例】66-70
本书《趣学C语言和数据结构100例》精选了5个典型的数据结构问题及C语言实现,涵盖链表与数组操作,如有序集合的集合运算、有序序列表的合并、数组中两顺序表位置互换、三递增序列公共元素查找及奇偶数重排。通过详细解析与代码示例,帮助读者深入理解数据结构与算法设计的核心思想,提升编程技能。
32 4
|
21天前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
43 5
|
20天前
|
并行计算 算法 测试技术
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面,旨在通过综合策略提升程序性能,满足实际需求。
49 1
|
存储 缓存 C语言
数据结构——双链表(C语言)
数据结构——双链表(C语言)