【C语言 - 数据结构】树、二叉树(下篇)(上)

简介: 【C语言 - 数据结构】树、二叉树(下篇)

一、二叉树的遍历原理


1.1原理:


二叉树的遍历(traveing binary tree)是指从根结点出发,按照某种次序依次访问二叉树中所有的结点,使每个结点都被访问一次,且仅被访问一次。


这里有两个关键词:访问和次序。


1.2.1访问


访问其实是要根据实际的需要来确定具体做什么,比如对每个结点进行相关计算,输出打印等,它算作是一个抽象操作。在这里我们可以简单地假定就是输出结点的数据信息。


1.2.2次序


二叉树的遍历次序不同于线性结构,最多也就是从头至尾、循环、双向等简单的遍历方式。树的结点之间不存在唯一的前驱和后继关系,在访问一个结点后,下一个被访问的结点面临着不同的选择就像你人生的道路上,高考填志愿要面临哪个城市、哪所大学、具体专业等选择,由选择方式的不同,遍历的次序就完全不同了。


1669440309505.jpg


二、二叉树的前序、中序、后序遍历


2.1二叉树遍历的几种方式


二叉树的遍历方式可以很多,如果我们限制了从左到右的习惯方式,那么主要就分为四种:


前序遍历、中序遍历、后序遍历、层次遍历。


这四种遍历方式的基本顺序和在数组中存储的形式如下图所示:


1669440338701.jpg


2.2前序遍历


规则是若 叉树为空,则空操作返回,否则先访问根结点,然后前序遍历左子树,再前序遍历右子树。如图遍历的顺序为: ABDGHCEIF


1669440355197.jpg


步骤:1、先造一颗树


造树:

#include<stdio.h>
#include<stdlib.h>
typedef int BTDataType;
typedef struct BinaryTreeNode
{
       struct BinaryTreeNode* left;
       struct BinaryTreeNode* right;
       BTDataType data;
}BTNode;
malloc一块空间
BTNode* BuyBTNode(BTDataType x)
{
       BTNode* node = (BTNode*)malloc(sizeof(BTNode));
       if (node == NULL)
       {
              printf("malloc fail\n");
              exit(-1);
       }
       node->data = x;
       node->left = node->right = NULL;
       return node;
}

紧接着实现链接

BTNode* CreatBinaryTree()
{
       BTNode* node1 = BuyBTNode(1);
       BTNode* node2 = BuyBTNode(2);
       BTNode* node3 = BuyBTNode(3);
       BTNode* node4 = BuyBTNode(4);
       BTNode* node5 = BuyBTNode(5);
       BTNode* node6 = BuyBTNode(6);
       node1->left = node2;
       node1->right = node4;
       node2->left = node3;
       node4->left = node5;
       node4->right = node6;
       return node1;
}


2、写前序遍历和main函数

void PrevOrder(BTNode* root)//前序遍历
{
       if (root == NULL)//如果根是空就return
       {
              printf("NULL ");
              return;
       }
       printf("%d ", root->data);
       PrevOrder(root->left);//左子树
       PrevOrder(root->right);//右子树
}
int main()
{
       BTNode* tree = CreatBinaryTree();
       PrevOrder(tree);
       return 0;
}

程序运行结果 :(对照1、2、3、4、5、6)上图

1669440406318.jpg


2.3中序遍历


规则是若树为空,则空操作返回,否则从根结点开始(注意并不是先访问根结 点) ,中序遍历根结点的左子树,然后是访问根结点,最后中序遍历右子树 如图所示, 遍历的顺序为GDHBAELCF.


1669440423329.jpg

void InOrder(BTNode* root)//中序遍历
{
       if (root == NULL)//如果根是空就return
       {
              printf("NULL ");
              return;
       }
       InOrder(root->left);//先左子树
       printf("%d ", root->data);
       InOrder(root->right);//再右子树
}
int main()
{
       BTNode* tree = CreatBinaryTree();
       InOrder(tree);
       return 0;
}

程序运行结果

1669440444655.jpg


2.4后序遍历


规则是若树为空,则空操作返回,否则从左到右先叶子后结点的方式遍历访问左右子树,最后是访问根结点 如图所示 遍历的顺序为 GHDBIEFCA。


1669440460875.jpg

void BackOrder(BTNode* root)
{
       if (root == NULL)//如果根是空就return
       {
              printf("NULL ");
              return;
       }
       BackOrder(root->left);
       BackOrder(root->right);
       printf("%d ", root->data);
}
int main()
{
       BTNode* tree = CreatBinaryTree();
       BackOrder(tree);
       return 0;
}

程序运行结果:


1669440481034.jpg


二叉树的遍历的几种路径 (小结):网上找的图


1669440490915.jpg


2.5层序遍历


层序遍历:除了先序遍历、中序遍历、后序遍历外,还可以对二叉树进行层序遍历。设二叉树的根节点所在 层数为1,层序遍历就是从所在二叉树的根节点出发,首先访问第一层的树根节点,然后从左到右访问第2层 上的节点,接着是第三层的节点,以此类推,自上而下,自左至右逐层访问树的结点的过程就是层序遍历。


1669440509937.jpg

1669440517714.jpg

层序遍历与之前的三种遍历情况有所不同,层序遍历的实现依赖于队列,会比较麻烦一些。


1669440527619.jpg


//层序遍历
void LevelOrder(BTNode* root)
{
       Queue q;
       QueueInit(&q);
       if (root)
       {
              QueuePush(&q, root);//先插入根节点
       }
       while (QueueEmpty(&q))
       {
              BTNode* front = QueueFront(&q);
              QueuePop(&q);//把队列里根节点的指针拿出来,但是指针指向的节点的值没有被销毁;
              printf("%d ", front->data);
              if (front->left)
              {
                      QueuePush(&q, front->left);
              }
              if (front->right)
              {
                      QueuePush(&q, front->right);
              }
              printf("\n");
       }
       QueueDestry(&q);
}

三、二叉树的拓展


3.1计算树的结点的个数


low版(代码比较挫)但是容易理解

int count = 0;
void BTreeSize(BTNode* root)
{
       if (root == NULL)
              return;
       ++count;
       BTreeSize(root->left);
       BTreeSize(root->right);
       //后序
}

注意:


这里为什么不用static静态变量。


因为静态变量在静态区,是整个程序结束后才销毁,而且局部静态变量不能置零


所以如果再计算下一个树的结点就会和上一个树累加。


static只初始化一次,所以要么就是全局静态变量。


具体的调用方法://更好的计数方法,既不使用全局,也不使用静态变量-

//思想遍历加计数(传地址调用)指针

void BTreeSize(BTNode* root, int* pCount)
{
       if (root == NULL)
              return;
       ++(*pCount);//把一个变量的地址传过去
       BTreeSize(root->left, pCount);
       BTreeSize(root->right, pCount);
       //后序
}

3.2计算树的叶子结点的个数

1669440580892.jpg

思路:叶子结点的左右结点都为空,递归+分治思想


因此:代码如下


int BTreeLaafSize(BTNode* root)
{
       if (root == NULL)
              return 0;
       if (root->left == NULL && root->right == NULL)
              return 1;
       return BTreeLaafSize(root->left) + BTreeLaafSize(root->right);
}
相关文章
|
5月前
|
存储 C语言
c语言——二叉树
二叉树的概念在这里就不进行过多的赘述,那么主要说一下我认为重要的部分,第一点就是二叉树里面部分概念的理解:就比如说,你对于如何构建二叉树,掌握的十分深刻,但刷题的时候对于一些题目所给的概念不清楚,导致看不明白题目,这课不好,二叉树的概念如下图所示,其实都很简单,主要是当给他的名字时,你明不明白。还有对于满二叉树与完全二叉树。
109 0
|
8月前
|
存储 算法 Java
算法系列之数据结构-二叉树
树是一种重要的非线性数据结构,广泛应用于各种算法和应用中。本文介绍了树的基本概念、常见类型(如二叉树、满二叉树、完全二叉树、平衡二叉树、B树等)及其在Java中的实现。通过递归方法实现了二叉树的前序、中序、后序和层次遍历,并展示了具体的代码示例和运行结果。掌握树结构有助于提高编程能力,优化算法设计。
256 10
 算法系列之数据结构-二叉树
|
8月前
|
算法 Java
算法系列之数据结构-Huffman树
Huffman树(哈夫曼树)又称最优二叉树,是一种带权路径长度最短的二叉树,常用于信息传输、数据压缩等方面。它的构造基于字符出现的频率,通过将频率较低的字符组合在一起,最终形成一棵树。在Huffman树中,每个叶节点代表一个字符,而每个字符的编码则是从根节点到叶节点的路径所对应的二进制序列。
206 3
 算法系列之数据结构-Huffman树
|
8月前
|
存储 自然语言处理 数据库
【数据结构进阶】AVL树深度剖析 + 实现(附源码)
在深入探讨了AVL树的原理和实现后,我们不难发现,这种数据结构不仅优雅地解决了传统二叉搜索树可能面临的性能退化问题,还通过其独特的平衡机制,确保了在任何情况下都能提供稳定且高效的查找、插入和删除操作。
635 19
|
10月前
|
存储 C++
【C++数据结构——树】哈夫曼树(头歌实践教学平台习题) 【合集】
【数据结构——树】哈夫曼树(头歌实践教学平台习题)【合集】目录 任务描述 相关知识 测试说明 我的通关代码: 测试结果:任务描述 本关任务:编写一个程序构建哈夫曼树和生成哈夫曼编码。 相关知识 为了完成本关任务,你需要掌握: 1.如何构建哈夫曼树, 2.如何生成哈夫曼编码。 测试说明 平台会对你编写的代码进行测试: 测试输入: 1192677541518462450242195190181174157138124123 (用户分别输入所列单词的频度) 预
328 14
【C++数据结构——树】哈夫曼树(头歌实践教学平台习题) 【合集】
|
9月前
|
定位技术 C语言
c语言及数据结构实现简单贪吃蛇小游戏
c语言及数据结构实现简单贪吃蛇小游戏
|
10月前
|
搜索推荐 C语言
数据结构(C语言)之对归并排序的介绍与理解
归并排序是一种基于分治策略的排序算法,通过递归将数组不断分割为子数组,直到每个子数组仅剩一个元素,再逐步合并这些有序的子数组以得到最终的有序数组。递归版本中,每次分割区间为[left, mid]和[mid+1, right],确保每两个区间内数据有序后进行合并。非递归版本则通过逐步增加gap值(初始为1),先对单个元素排序,再逐步扩大到更大的区间进行合并,直至整个数组有序。归并排序的时间复杂度为O(n*logn),空间复杂度为O(n),且具有稳定性,适用于普通排序及大文件排序场景。
|
10月前
|
Java C++
【C++数据结构——树】二叉树的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现二叉树的基本运算。​ 相关知识 创建二叉树 销毁二叉树 查找结点 求二叉树的高度 输出二叉树 //二叉树节点结构体定义 structTreeNode{ intval; TreeNode*left; TreeNode*right; TreeNode(intx):val(x),left(NULL),right(NULL){} }; 创建二叉树 //创建二叉树函数(简单示例,手动构建) TreeNode*create
276 12
|
10月前
|
C++
【C++数据结构——树】二叉树的性质(头歌实践教学平台习题)【合集】
本文档介绍了如何根据二叉树的括号表示串创建二叉树,并计算其结点个数、叶子结点个数、某结点的层次和二叉树的宽度。主要内容包括: 1. **定义二叉树节点结构体**:定义了包含节点值、左子节点指针和右子节点指针的结构体。 2. **实现构建二叉树的函数**:通过解析括号表示串,递归地构建二叉树的各个节点及其子树。 3. **使用示例**:展示了如何调用 `buildTree` 函数构建二叉树并进行简单验证。 4. **计算二叉树属性**: - 计算二叉树节点个数。 - 计算二叉树叶子节点个数。 - 计算某节点的层次。 - 计算二叉树的宽度。 最后,提供了测试说明及通关代
188 10
|
存储 缓存 C语言
数据结构——双链表(C语言)
数据结构——双链表(C语言)

热门文章

最新文章