C语言数据结构(16)--二叉树的层序遍历代码实现

简介: 本文目录1. 背景2. 思路3. 代码实现

1. 背景

在上一篇中,我们利用递归很轻易的就实现了二叉树的前序、中序、后续遍历,但是层序遍历仅仅利用递归貌似是解决不了的。

image.png


在如上图的树中,我们如何先从上至下,然后从左至右的按层次进行遍历,即A-B-C-D-E-F-G这样的顺序呢。


2. 思路

每次在访问下一层次节点之前,应该将上一级节点输出,而上一级节点无疑从层次上先于下一级,我们联想到先进先出的队列模型,我们可以利用一个队列,在递归访问二叉树下一层次之前,将本层次的节点放入队列,这样就实现了输出时,也是按层次先后输出的。


之前有一版本写错了,层序遍历应该递归访问队列,每次输出一个层级,而不是递归二叉树。

感谢才情a指出问题,他的博客附上表示敬意

2020.2.25修正一版


3. 代码实现

#include<stdio.h>
#define QUEUE_LENGTH 100 //队列最大元素个数
/*
* 二叉树的层序遍历演示DEMO
* 作者:熊猫大大
* 时间:2019-12-15
*/
//二叉树结构体
typedef struct {
  char data;//数据区域(为了保存ABCD,直接用char当做数据域,便于和文章中的插图对应,稳!)
  struct BinaryTreeNode* left;//左子节点
  struct BinaryTreeNode* right;//右子节点
}BinaryTreeNode;
//为树的当前节点添加左子节点
int addLeftChild(BinaryTreeNode* curNode, char leftData)
{
  //分配新节点
  BinaryTreeNode* leftNode = (BinaryTreeNode*)malloc(sizeof(BinaryTreeNode));
  //为新节点挂载数据
  leftNode->data = leftData;
  //新节点暂时无子节点
  leftNode->left = NULL;
  leftNode->right = NULL;
  //将新节点挂到当前节点下
  curNode->left = leftNode;
  return 1;
}
//为树的当前节点添加右子节点
int addRightChild(BinaryTreeNode* curNode, char rightData)
{
  //分配新节点
  BinaryTreeNode* rightNode = (BinaryTreeNode*)malloc(sizeof(BinaryTreeNode));
  //为新节点挂载数据
  rightNode->data = rightData;
  //新节点暂时无子节点
  rightNode->left = NULL;
  rightNode->right = NULL;
  //将新节点挂到当前节点下
  curNode->right = rightNode;
  return 1;
}
// 队列结构体
struct Queue
{
  BinaryTreeNode element[QUEUE_LENGTH];//队列元素应该是节点指针
  int head;//头部位置
  int tail;//尾部位置
};
//入队
int queueIn(struct Queue* p, BinaryTreeNode *node)
{
  //满了
  if (p->tail>QUEUE_LENGTH - 1)
  {
    return 0;//入队失败
  }
  p->element[p->tail] = *node;
  p->tail++;
  return 1;
}
//出队
int queueOut(struct Queue* p, BinaryTreeNode *node)
{
  if (p->head == p->tail)
    return 0;//出队失败
  *node = p->element[p->head++];
  return 1;
}
// 层序遍历
void leverOrder(struct Queue* queue)
{
  struct Queue newQueue;
  newQueue.head = 0;
  newQueue.tail = 0;
  //上一级出队
  BinaryTreeNode *result=(BinaryTreeNode*)malloc(sizeof(BinaryTreeNode));;
  while (queueOut(queue, result) == 1) {
    printf("%c", result->data);
    //将子节点先加入队列,后续再遍历更深的层次
    if (result->left != NULL) {
      queueIn(&newQueue, result->left);
    }
    if (result->right != NULL) {
      queueIn(&newQueue, result->right);
    }
  }
  leverOrder(&newQueue);
}
//----------------------------------------------------------------------------------------------------测试入口区域
int main()
{
  //初始化队列
  struct Queue queue;
  queue.head = 0;
  queue.tail = 0;
  //设定根节点
  BinaryTreeNode root;
  //根节点A
  root.data = 'A';
  addLeftChild(&root, 'B');
  addRightChild(&root, 'C');
  //为B节点增加子节点
  addLeftChild(root.left, 'D');
  addRightChild(root.left, 'E');
  //为C节点增加子节点
  addLeftChild(root.right, 'F');
  addRightChild(root.right, 'G');
  //为D节点增加子节点
  BinaryTreeNode *b = root.left;
  addLeftChild(b->left, 'H');
  addRightChild(b->left, 'I');
  //为E节点增加子节点
  addRightChild(b->right, 'K');
  //为F节点增加子节点
  BinaryTreeNode *c = root.right;
  addLeftChild(c->left, 'L');
  addRightChild(c->left, 'M');
  printf("\n层序遍历:");
  //根节点第一个进入队列
  queueIn(&queue, &root);
  leverOrder(&queue);
  return 1;
}
相关文章
|
3月前
|
算法 数据处理 C语言
C语言中的位运算技巧,涵盖基本概念、应用场景、实用技巧及示例代码,并讨论了位运算的性能优势及其与其他数据结构和算法的结合
本文深入解析了C语言中的位运算技巧,涵盖基本概念、应用场景、实用技巧及示例代码,并讨论了位运算的性能优势及其与其他数据结构和算法的结合,旨在帮助读者掌握这一高效的数据处理方法。
91 1
|
2天前
|
定位技术 C语言
c语言及数据结构实现简单贪吃蛇小游戏
c语言及数据结构实现简单贪吃蛇小游戏
|
20天前
|
搜索推荐 C语言
数据结构(C语言)之对归并排序的介绍与理解
归并排序是一种基于分治策略的排序算法,通过递归将数组不断分割为子数组,直到每个子数组仅剩一个元素,再逐步合并这些有序的子数组以得到最终的有序数组。递归版本中,每次分割区间为[left, mid]和[mid+1, right],确保每两个区间内数据有序后进行合并。非递归版本则通过逐步增加gap值(初始为1),先对单个元素排序,再逐步扩大到更大的区间进行合并,直至整个数组有序。归并排序的时间复杂度为O(n*logn),空间复杂度为O(n),且具有稳定性,适用于普通排序及大文件排序场景。
|
1月前
|
Java C++
【C++数据结构——树】二叉树的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现二叉树的基本运算。​ 相关知识 创建二叉树 销毁二叉树 查找结点 求二叉树的高度 输出二叉树 //二叉树节点结构体定义 structTreeNode{ intval; TreeNode*left; TreeNode*right; TreeNode(intx):val(x),left(NULL),right(NULL){} }; 创建二叉树 //创建二叉树函数(简单示例,手动构建) TreeNode*create
48 12
|
1月前
|
C++
【C++数据结构——树】二叉树的性质(头歌实践教学平台习题)【合集】
本文档介绍了如何根据二叉树的括号表示串创建二叉树,并计算其结点个数、叶子结点个数、某结点的层次和二叉树的宽度。主要内容包括: 1. **定义二叉树节点结构体**:定义了包含节点值、左子节点指针和右子节点指针的结构体。 2. **实现构建二叉树的函数**:通过解析括号表示串,递归地构建二叉树的各个节点及其子树。 3. **使用示例**:展示了如何调用 `buildTree` 函数构建二叉树并进行简单验证。 4. **计算二叉树属性**: - 计算二叉树节点个数。 - 计算二叉树叶子节点个数。 - 计算某节点的层次。 - 计算二叉树的宽度。 最后,提供了测试说明及通关代
46 10
|
1月前
|
存储 算法 测试技术
【C++数据结构——树】二叉树的遍历算法(头歌教学实验平台习题) 【合集】
本任务旨在实现二叉树的遍历,包括先序、中序、后序和层次遍历。首先介绍了二叉树的基本概念与结构定义,并通过C++代码示例展示了如何定义二叉树节点及构建二叉树。接着详细讲解了四种遍历方法的递归实现逻辑,以及层次遍历中队列的应用。最后提供了测试用例和预期输出,确保代码正确性。通过这些内容,帮助读者理解并掌握二叉树遍历的核心思想与实现技巧。
49 2
|
2月前
|
数据库
数据结构中二叉树,哈希表,顺序表,链表的比较补充
二叉搜索树,哈希表,顺序表,链表的特点的比较
数据结构中二叉树,哈希表,顺序表,链表的比较补充
|
3月前
|
C语言
【数据结构】栈和队列(c语言实现)(附源码)
本文介绍了栈和队列两种数据结构。栈是一种只能在一端进行插入和删除操作的线性表,遵循“先进后出”原则;队列则在一端插入、另一端删除,遵循“先进先出”原则。文章详细讲解了栈和队列的结构定义、方法声明及实现,并提供了完整的代码示例。栈和队列在实际应用中非常广泛,如二叉树的层序遍历和快速排序的非递归实现等。
328 9
|
3月前
|
存储 算法
非递归实现后序遍历时,如何避免栈溢出?
后序遍历的递归实现和非递归实现各有优缺点,在实际应用中需要根据具体的问题需求、二叉树的特点以及性能和空间的限制等因素来选择合适的实现方式。
53 1
|
1月前
|
存储 C语言 C++
【C++数据结构——栈与队列】顺序栈的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现顺序栈的基本运算。开始你的任务吧,祝你成功!​ 相关知识 初始化栈 销毁栈 判断栈是否为空 进栈 出栈 取栈顶元素 1.初始化栈 概念:初始化栈是为栈的使用做准备,包括分配内存空间(如果是动态分配)和设置栈的初始状态。栈有顺序栈和链式栈两种常见形式。对于顺序栈,通常需要定义一个数组来存储栈元素,并设置一个变量来记录栈顶位置;对于链式栈,需要定义节点结构,包含数据域和指针域,同时初始化栈顶指针。 示例(顺序栈): 以下是一个简单的顺序栈初始化示例,假设用C语言实现,栈中存储
142 77

热门文章

最新文章