二叉树遍历——递归链式(C语言实现)(上)

简介: 二叉树遍历——递归链式(C语言实现)

前,中,后序遍历

首先我们定义一个结构体,链式储存,那么肯定有一个左孩子和右孩子,自身也要储存值。

typedef char BTDataType;//重命名,方便更改类型
typedef struct BinaryTreeNode
{
  BTDataType _data;//自身储存值
  struct BinaryTreeNode* _left;//左孩子
  struct BinaryTreeNode* _right;//右孩子
}BTNode;

先创建如下的二叉树:

如果二叉树是这种情况,前中后怎么进行遍历呢?

前序遍历:

前序是先访问根节点,再访问左子树,最后访问右子树。(这里要注意,B是A的左子树,C是A的右子树,D是B的左子树,以此类推)

遍历都是从根节点进入的,那么我们第一个访问的肯定是A,然后访问的是结点B,正常来说又要访问结点的C了,但是B结点也有子孙,所以要先访问B的所有子孙才能访问C的子孙。

递归到D结点之后,D就是根节点,两边的空指针就是左右孩子,先进入左孩子,因为是空指针,所以返回到D,再进行右孩子的访问,右孩子也是个空指针,那么也返回到D,D的所有子孙都访问完之后返回B, 然后又要访问B的右边的子孙(也是右树)。

那么顺序就是:A->B->D->NULL->NULL-> E->G->NULL->NULL->NULL->C->F->H->NULL->NULL->I->NULL->NULL->NULL

代码实现:

void BinaryTreePrevOrder(BTNode* root)//最初传入的就是祖先结点A
{
  if (root == NULL)//如果遇到空结点就返回
  {
    printf("NULL ");
    return;
  }
  printf("%c ", root->_data);//打印每个结点中的内容,也等于访问该节点
  BinaryTreePrevOrder(root->_left);//进入左
  BinaryTreePrevOrder(root->_right);//进入右
}

先写一个函数来测试一下这段函数有没有作用:

void test1()
{
  BTNode n1;
  BTNode n2;
  BTNode n3;
  BTNode n4;
  BTNode n5;
  BTNode n6;
  BTNode n7;
  BTNode n8;
  BTNode n9;
  n1._data = 'A';
  n1._left = &n2;
  n1._right = &n3;
  n2._data = 'B';
  n2._left = &n4;
  n2._right =&n5;
  n3._data = 'C';
  n3._left = &n6;
  n3._right = NULL;
  n4._data = 'D';
  n4._left = NULL;
  n4._right = NULL;
  n5._data = 'E';
  n5._left = &n7;
  n5._right = NULL;
  n6._data = 'F';
  n6._left = &n8;
  n6._right = &n9;
  n7._data = 'G';
  n7._left = NULL;
  n7._right = NULL;
  n8._data = 'H';
  n8._left = NULL;
  n8._right = NULL;
  n9._data = 'I';
  n9._left = NULL;
  n9._right = NULL;
  BinaryTreePrevOrder(&n1);
}
int main()
{
  test1();
  return 0;
}

结果没错。

刚调用这个函数的时候传入的就是祖先节点,不为空就直接打印,然后进入左子树,和右子树,不为空就打印,是空就返回,返回的时候就在原来调用函数的位置,这里最重要的就是顺序。

中序遍历

中序遍历是,先访问左子树,再访问根,最后访问右子树。

知道前序遍历就好办了,那么这里调整一下递归的顺序就好了。

void BinaryTreeInOrder(BTNode* root)//左 根 右
{
  if (root == NULL)
  {
    printf("NULL ");
    return;
  }
  BinaryTreeInOrder(root->_left);
  printf("%c ", root->_data);
  BinaryTreeInOrder(root->_right);
}

递归到D结点的时候再进入D的左子树,发现是空指针返回,然后返回到D的位置在访问D,最后再进行D右子树的访问。

因为一开始并没有进行打印的操作,所以在进入D左边的空指针之前就没有打印途中的A B D,这就是顺序的重要性。

后序遍历

后序遍历是:先访问左,在访问右,在访问根。

这里只要把打印的事情放到最后就好了。

void BinaryTreePostOrder(BTNode* root)//左 右 根
{
  if (root == NULL)
  {
    printf("NULL ");
    return;
  }
  BinaryTreePostOrder(root->_left);//还是先从左开始访问,一直到底
  BinaryTreePostOrder(root->_right);
  printf("%c ", root->_data);
}

递归到D的位置的时候,先进入D的左树,发现是空指针就返回,返回之后是在D的位置,这里一定不要打印,再进入D的右树,发现是空指针然后返回,这样D的左子树和右子树都访问完成了,最后在进行D的访问。

结点个数与叶子个数

结点个数

计算节点个数可以定义一个全局的静态变量,但是缺点很明显,每次计算完都要重新值置为零,很麻烦。

我们可以利用函数的返回值和递归解决这个问题,核心思路是这样的:

例:

总数应该是祖先结点A+左子树+右子树,左子树里面还有左子树+右子树,右子树里面还有左子树和右子树,就和上面的遍历差不多的逻辑。

int BinaryTreeSize(BTNode* root)
{
  if (root == NULL)
  {
    return 0;//如果到了空指针就返回0
  }
  return BinaryTreeSize(root->_left) + 
    BinaryTreeSize(root->_right) + 1;//没到空指针就记住这个结点,并且知道找到空指针为止
}

如果找到不是空的结点就用+1记住该节点,遇到空结点就返回0。

这样就不会出现静态全局变量需要重置的问题了。

叶子数量

计算叶子的数量,就要找叶子节点的特点,叶子的特点是孩子节点都是空。

例:

B,D结点的两个孩子都是空,所以是叶子节点,那么代码实现只需要判断一下就可以了。

int BinaryTreeLeafSize(BTNode* root)
{
  if (root == NULL)
  {
    return 0;
  }
  if (root->_left == NULL && root->_right == NULL)
  {
    return 1;//如果孩子都为空就返回1,说明这个结点是叶子节点
  }
  return BinaryTreeLeafSize(root->_left) + BinaryTreeLeafSize(root->_right);//调用左子树和右子树
}

求第k层的结点个数与树的高度

求k层的节点个数

例:

想访问这棵树的第三层,那么这层就等于左子树和右子树的第二层,也就等于k-1,那么直到k等于1,说明这里就是我们要访问的结点。

遇到空就返回0,遇到该层结点就返回1,比如说这棵树,A->B->NULL返回0到B的位置,B->D,到达该层数,返回1到B,然后到A的右子树进行访问。

int BinaryTreeLevelKSize(BTNode* root, int k)
{
  assert(k > 0);//不能是负数
  if (root == NULL)
  {
    return 0;
  }
  if (k == 1)
  {
    return 1;
  }
  return BinaryTreeLevelKSize(root->_left, k - 1) + BinaryTreeLevelKSize(root->_right, k - 1);
}

树的高度

例:

思路是,找A的左子树和右子树,最后比一比谁的更长,A的左子树最长的是D,长度为2,右子树最长的是C,长度为1,所以这棵树的高度为2。

int TreeHeight(BTNode* root)
{
  if (root == NULL)
  {
    return 0;
  }
  int x = TreeHeight(root->_left);
  int y = TreeHeight(root->_right);
  return x > y ? x + 1: y + 1;//+1是记录非空结点
}

相关文章
|
2月前
|
存储 算法 C语言
C语言中常见的字符串处理技巧,包括字符串的定义、初始化、输入输出、长度计算、比较、查找与替换、拼接、截取、转换、遍历及注意事项
本文深入探讨了C语言中常见的字符串处理技巧,包括字符串的定义、初始化、输入输出、长度计算、比较、查找与替换、拼接、截取、转换、遍历及注意事项,并通过案例分析展示了实际应用,旨在帮助读者提高编程效率和代码质量。
113 4
|
2月前
|
C语言
【数据结构】二叉树(c语言)(附源码)
本文介绍了如何使用链式结构实现二叉树的基本功能,包括前序、中序、后序和层序遍历,统计节点个数和树的高度,查找节点,判断是否为完全二叉树,以及销毁二叉树。通过手动创建一棵二叉树,详细讲解了每个功能的实现方法和代码示例,帮助读者深入理解递归和数据结构的应用。
147 8
|
3月前
|
机器学习/深度学习 C语言
【c语言】一篇文章搞懂函数递归
本文详细介绍了函数递归的概念、思想及其限制条件,并通过求阶乘、打印整数每一位和求斐波那契数等实例,展示了递归的应用。递归的核心在于将大问题分解为小问题,但需注意递归可能导致效率低下和栈溢出的问题。文章最后总结了递归的优缺点,提醒读者在实际编程中合理使用递归。
83 7
|
3月前
|
C语言
c语言回顾-函数递归(上)
c语言回顾-函数递归(上)
51 2
|
3月前
|
C语言
链式顺序表实现(C语言描述)
本文介绍了如何在C语言中实现链式顺序表,包括数据结构的定义、节点的创建、数据的插入和删除以及链表的打印和销毁。
52 2
|
3月前
|
C语言
c语言回顾-函数递归(下)
c语言回顾-函数递归(下)
52 0
|
4月前
|
存储 算法 C语言
C语言手撕实战代码_二叉树_构造二叉树_层序遍历二叉树_二叉树深度的超详细代码实现
这段代码和文本介绍了一系列二叉树相关的问题及其解决方案。其中包括根据前序和中序序列构建二叉树、通过层次遍历序列和中序序列创建二叉树、计算二叉树节点数量、叶子节点数量、度为1的节点数量、二叉树高度、特定节点子树深度、判断两棵树是否相似、将叶子节点链接成双向链表、计算算术表达式的值、判断是否为完全二叉树以及求二叉树的最大宽度等。每道题目均提供了详细的算法思路及相应的C/C++代码实现,帮助读者理解和掌握二叉树的基本操作与应用。
|
1月前
|
存储 C语言 开发者
【C语言】字符串操作函数详解
这些字符串操作函数在C语言中提供了强大的功能,帮助开发者有效地处理字符串数据。通过对每个函数的详细讲解、示例代码和表格说明,可以更好地理解如何使用这些函数进行各种字符串操作。如果在实际编程中遇到特定的字符串处理需求,可以参考这些函数和示例,灵活运用。
66 10
|
1月前
|
存储 程序员 C语言
【C语言】文件操作函数详解
C语言提供了一组标准库函数来处理文件操作,这些函数定义在 `<stdio.h>` 头文件中。文件操作包括文件的打开、读写、关闭以及文件属性的查询等。以下是常用文件操作函数的详细讲解,包括函数原型、参数说明、返回值说明、示例代码和表格汇总。
52 9
|
1月前
|
存储 Unix Serverless
【C语言】常用函数汇总表
本文总结了C语言中常用的函数,涵盖输入/输出、字符串操作、内存管理、数学运算、时间处理、文件操作及布尔类型等多个方面。每类函数均以表格形式列出其功能和使用示例,便于快速查阅和学习。通过综合示例代码,展示了这些函数的实际应用,帮助读者更好地理解和掌握C语言的基本功能和标准库函数的使用方法。感谢阅读,希望对你有所帮助!
41 8