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

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

查找值为x的结点与层序遍历

查找值为x的结点

查找整棵树中的储存的值为x的结点首先需要遍历,然后判断哪个结点是我们要找的结点, 不过返回的时候需要进行判断,不然会出现这种情况:

找D的时候,从A的左子树开始找,找不到返回空,找到了返回该节点,但是返回该节点的时候回到的位置是上一个结点的位置,如果没有判断就会去下个树中去找,并且不会将该节点返回到我们需要的地方。

如果加一个判断,顺利的返回就好了。

BTNode* BinaryTreeFind(BTNode* root, BTDataType x)
{
  if (root == NULL)
  {
    return NULL;
  }
  if (root->_data == x)
  {
    return root;
  }
  BTNode* x1 =BinaryTreeFind(root->_left, x);//找左子树
  if (x1)//判断是否为空,空是找到了,非空是没找到
  {
    return x1;//找到了就返回找到的结点,位置是上一层的找左子树的函数
  }
  BTNode* y =BinaryTreeFind(root->_right , x);
  if (y)
  {
    return y;
  }
  return NULL;
}

这里还可以进行修改值。

层序遍历

层序遍历是一层一层的进行访问:

从祖先结点开始,遇到空指针返回。

那么怎么才能把所有的都访问到呢?我们需要借助队列:

在队中的头结点出队的时候,会将左子树和右子树进行入队操作,如果左子树和右子树中有空指针将不会进行入队操作。

#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
#include <stdbool.h>
typedef char BTDataType;
typedef struct BinaryTreeNode
{
  BTDataType _data;
  struct BinaryTreeNode* _left;
  struct BinaryTreeNode* _right;
}BTNode;
typedef BTNode* SD;//将队列中的储存的内容改成二叉树结点类型
typedef struct QListNode
{
  struct QListNode* next;
  SD data;
}QL;
typedef struct Queue
{
  QL* head;//头结点
  QL* tail;//尾结点
  int siz;
}Qu;
void QueueInit(Qu* q);//初始化
void QueuePush(Qu* q, SD x);//入队
void QueuePop(Qu* q);//出队
SD QueueFront(Qu* q);//获取队头元素
SD QueueBack(Qu* q);//获取队尾元素
int QueueSize(Qu* q);//获取队列中有效元素个数
bool QueueEmpty(Qu* q);//检测队列是否为空
void QueueDestroy(Qu* q);//销毁队列
#include "queue.h"
void QueueInit(Qu* q)//初始化
{
  assert(q);
  q->head = q->tail = NULL;
  q->siz = 0;
}
void QueueDestroy(Qu* q)//销毁队列 
{
  assert(q);
  QL* cur = q->head;
  while (cur)
  {
    QL* del = cur -> next;
    free(cur);
    cur = del;
  }
  q->head = q->tail = NULL;
  q->siz = 0;
}
void QueuePush(Qu* q, SD x)//入队
{
  assert(q);
  QL* w = (QL*)malloc(sizeof(Qu));
  if (w == NULL)
  {
    perror("malloc tail");
    exit(-1);
  }
  else
  {
    w->data = x;
    w->next = NULL;
  }
  if (q->head == NULL)
  {
    q->head = q->tail = w;
  }
  else
  {
    q->tail->next = w;
    q->tail = w;
  }
  q->siz++;
}
bool QueueEmpty(Qu* q)//判断
{
  assert(q);
  return q->head == NULL;
}
void QueuePop(Qu* q)//出队
{
  assert(q);
  assert(!QueueEmpty(q));
  if (q->head == NULL)//防止tail成为野指针
  {
    free(q->head);
    q->head = q->tail = NULL;
  }
  QL* cur = q->head;
  q->head = q->head->next;
  free(cur);
  cur = NULL;
  q->siz--;
}
SD QueueFront(Qu* q)//获取队头元素
{
  assert(q);
  assert(!QueueEmpty(q));
  return q->head->data;
}
SD QueueBack(Qu* q)//获取队尾元素
{
  assert(q);
  assert(!QueueEmpty(q));
  return q->tail->data;
}
int QueueSize(Qu* q)//获取队列中有效元素个数
{
  assert(q);
  return q->siz;
}
#include "queue.h"
void BinaryTreeLevelOrder(BTNode* root)
{
  Qu q;
  QueueInit(&q);//初始化队列
  if (root)//空指针不能入队
  {
    QueuePush(&q, root);//将结点存入队列中
  }
  while (!QueueEmpty(&q))//如果为空就不要进行入队和出队的操作了
  {
    BTNode* Front = QueueFront(&q);//获取队头元素
    printf("%c ", Front->_data);//进行打印
    QueuePop(&q);//弹出队头元素
    if (Front->_left)//判断左子树是不是空指针
      QueuePush(&q, Front->_left);
    if (Front->_right)//判断右子树是不是空指针
      QueuePush(&q, Front->_right);
  }
  printf("\n");
  QueueDestroy(&q);//销毁队列
}

销毁二叉树与判断二叉树是否为完全二叉树

销毁二叉树

销毁树的逻辑也是遍历,然后从底部销毁。

void BinaryTreeDestory(BTNode* root)
{
  if (root == NULL)
  {
    return;//找到底部返回上一层进行释放就可以了
  }
  BinaryTreeDestory(root->_left);//这里就是先从左子树开始
  BinaryTreeDestory(root->_right);
  free(root);
}

判断是否为完全二叉树

想判断二叉树是否为一个完全二叉树,就用刚才说的层序遍历:

例:

层序遍历很好查看:

  1. 当遇到空指针的时候,这一层后面的结点必须都是空指针,
  2. 下面的一层也必须都是空指针。

向上面的这种肯定不是,至少要吧C的左子树换成空指针,或者是B和C的右子树不是空指针,但是他们右子树的右子树必须是空指针。

这样的话,和层序遍历没啥区别,但是也有,因为我们这里遇到空指针也要入队,不然无法判断下一层是不是空指针。

因为A出队B C才会入队,B C出队,他们的子树才能入队,D出队的时候,他的子树也如对了(红色的),这样看来如果E结点是个空结点也不用担心最后一层的NULL不在队中。

当D出队后,下一个访问的就是空指针, 这时候,后面的所有结点都必须是空指针才行,不是就说明是非完全二叉树。

int BinaryTreeComplete(BTNode* root)
{
  Qu q;
  QueueInit(&q);
  QueuePush(&q, root);
  while (!QueueEmpty(&q))//还是正常的层序遍历操作
  {
    BTNode* Front = QueueFront(&q);
    if (Front == NULL)
    {
      break;//这里如果空指针是对头,就跳出进行入队的操作
    }
    QueuePop(&q);
    QueuePush(&q, Front->_left);
    QueuePush(&q, Front->_right);
  }
  while (!QueueEmpty(&q))//判断空指针
  {
    BTNode* Front = QueueFront(&q);
    QueuePop(&q);
    if (Front != NULL)//如果队中遇到的不是空指针,那么就不是完全二叉树
    {
      QueueDestroy(&q);
      return false;
    }
  }
  QueueDestroy(&q);
  return true;
}

显然我创建的的并不是完全二叉树。

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