欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 新闻 > 焦点 > 数据结构C语言描述9(图文结合)--二叉树和特殊书的概念,二叉树“最傻瓜式创建”与前中后序的“递归”与“非递归遍历”

数据结构C语言描述9(图文结合)--二叉树和特殊书的概念,二叉树“最傻瓜式创建”与前中后序的“递归”与“非递归遍历”

2025/1/8 12:56:46 来源:https://blog.csdn.net/weixin_74085818/article/details/144949932  浏览:    关键词:数据结构C语言描述9(图文结合)--二叉树和特殊书的概念,二叉树“最傻瓜式创建”与前中后序的“递归”与“非递归遍历”

前言

  • 这个专栏将会用纯C实现常用的数据结构和简单的算法;
  • 有C基础即可跟着学习,代码均可运行;
  • 准备考研的也可跟着写,个人感觉,如果时间充裕,手写一遍比看书、刷题管用很多,这也是本人采用纯C语言实现的原因之一;
  • 欢迎收藏 + 关注,本人将会持续更新。

文章目录

  • 树相关概念
    • 什么是二叉树
      • 二叉树定义
      • 基本概念
      • 基本形态
      • 二叉树性质
        • 性质1
        • 性质2
        • 性质3
        • 性质4
        • 性质五
    • 特殊二叉树
      • 满二叉树
      • 完全二叉树
  • 二叉树创建与遍历
    • 树创建
    • 树遍历
      • 前序
        • 递归
        • 非递归
      • 中序
        • 递归
        • 非递归
      • 后序
        • 递归
        • 非递归
    • 总代码

树相关概念

什么是二叉树

二叉树定义

二叉树是n (n≥0)个结点的有限集合

  • 每个节点最多有两个子节点,分别称为左子节点和右子节点。
  • 左子节点和右子节点可以为空
  • 二叉树的子树也是二叉树。

基本概念

  • 节点的度:一个节点含有的子树的个数称为该节点的度
  • 叶节点度为0的节点称为叶节点
  • 分支节点度不为0的节点,又称为非终端节点
  • 父节点:若一个节点含有子节点,则这个节点称为其子节点的父节点,又称为双亲结点
  • 子节点:一个节点含有的子树的根结点称为该节点的子节点,又称为孩子节点
  • 兄弟节点:具有相同父节点的节点互称为兄弟节点
  • 树的度:一棵树中,最大的节点的度称为树的度
  • 节点的层次从根节点开始定义,根为第1层,根的子节点为第2层,以此类推
  • 树的高度或深度:树中节点的最大层次
  • 堂兄弟节点双亲在同一层的节点互为堂兄弟
  • 节点的祖先:从根到该节点所经分支上的所有节点
  • 子孙:以某节点为根的子树中任一节点都称为该节点的子孙

基本形态

在这里插入图片描述

二叉树性质

性质1

二叉树第k (k>=1)层上至多有2k–1个结点。

思考题

若二叉树具有第k层的结点,那么第k层结点数的取值范围是多少?[1, 2k-1]

性质2

高度为h (h>=0)的二叉树至多有2h - 1个结点

思考题

高度为h的二叉树的结点数范围?

证明:高度为h的二叉树共有h层,第k层节点数范围为[1, 2k-1] ,故可知,高度为h的二叉树至少具有: ∑ k = 1 h 1 = h \sum _{k=1}^{h} 1=h k=1h1=h个节点;高度为h的二叉树至多具有 ∑ k = 1 h 2 k − 1 = 2 0 + 2 1 + . . . + 2 h − 1 = 2 h − 1 \sum ^{h}_{k=1}2^{k-1}=2^0+2^1+...+2^{h-1}=2^h-1 k=1h2k1=20+21+...+2h1=2h1
∑ k = 1 h 2 k − 1 = 2 0 + 2 1 + . . . + 2 h − 1 = 2 h − 1 \sum ^{h}_{k=1}2^{k-1}=2^0+2^1+...+2^{h-1}=2^h-1 k=1h2k1=20+21+...+2h1=2h1

性质3

设二叉树叶子结点数为n0,度为2的结点数为n2,则有:n0= n2+ 1。

性质4

结点数为n的完全二叉树的高度为: log ⁡ 2 n \log_2{n} log2n + 1或 log ⁡ 2 ( n + 1 ) \log_2{(n+1)} log2(n1)

性质五

给有n个结点的完全二叉树按层次编号,对于编号为i的结点:

  • 可计算i结点的双亲结点的编号
    • 若i = 1,则无双亲
    • 否则双亲编号为 i 2 \frac i 2 2i
  • 可计算i结点的左孩子结点的编号
    • 若2*i > n,则无左孩子
    • 否则其左孩子编号为2*i
  • 可计算i结点的右孩子结点的编号
    • 若2*i+1> n,则无右孩子
    • 否则其右孩子编号为2*i+1

特殊二叉树

满二叉树

  • 深度为h且具有2h-1个结点的二叉树
  • 每一层都容纳了该层所能容纳的最大结点数结点的二叉树
  • 没有度数为1的结点,且叶子结点均分布在最大层的二叉树

在这里插入图片描述

思考题

深度为h的满二叉树

  • 结点总数为? 2h - 1
  • 叶子结点数为? 2h-1
  • 度为1的结点数为?0
  • 度为2的结点数为?2h-1-1

具有n个结点的满二叉树

  • 有多少个叶子?(n + 1) / 2
  • 有多少个度为2的结点?(n - 1) / 2

完全二叉树

  • 除去最大层是一棵满二叉树
  • 除去最大层是一棵满二叉树

在这里插入图片描述

完全二叉树的几个非常有趣的特点:

  • 除去最大层是一棵满二叉树
  • 最大层上的结点向左充满
  • 叶子只可能分布在最大层和次大层上。
  • 度为1的结点至多有1个。
  • 一棵满二叉树一定是一棵完全二叉树,而一棵完全二叉树不一定是一棵满二叉树。
  • 对于具有相同结点数的二叉树而言,完全二叉树的高度一定是其中最小的

思考题

高度为h的完全二叉树的结点范围?[ 2k-1, 2k-1]

具有n个结点的二叉树的高度最小值为多少?高度最大值为多少? [ log ⁡ 2 n \log_2{n} log2n+1, n]

  • 完全二叉树最小: log ⁡ 2 n \log_2{n} log2n+1
  • 一层只有一个结点最大:n

二叉树创建与遍历

树创建

节点封装,二叉树,封装就需要封装两个孩子

typedef struct TreeNode {char data;struct TreeNode* LChild;struct TreeNode* RChild;
}TreeNode;TreeNode* create_node(char data)
{TreeNode* new_node = (TreeNode*)calloc(1, sizeof(TreeNode));assert(new_node);new_node->data = data;return new_node;
}

🚸 创建,这里采用“最傻瓜式”创建,如下代码所示:

// 傻瓜式建立树
void create_tree(TreeNode* parent, TreeNode* lChild, TreeNode* rChild)
{// 父亲不为 NULL 即可assert(parent);parent->LChild = lChild;parent->RChild = rChild;
}// 创建如下:
int main()
{// 一个一个创建节点TreeNode* a = create_node('a');   TreeNode* c = create_node('c');TreeNode* d = create_node('d');TreeNode* s = create_node('s');TreeNode* h = create_node('h');TreeNode* e = create_node('e');TreeNode* b = create_node('b');TreeNode* v = create_node('v');TreeNode* m = create_node('m');// 连接create_tree(a, c, d);   // a为跟节点create_tree(c, s, h);create_tree(d, e, b);create_tree(s, NULL, v);create_tree(h, NULL, NULL);create_tree(e, m, NULL);create_tree(b, NULL, NULL);return 0;
}

树遍历

前序

遍历顺序:左中右,动画如下所示(ppt制作):

在这里插入图片描述

递归
// 递归前序
void preorder_recursion(TreeNode* root)
{if (root == NULL) {printf("NULL ");return;}printf("%c ", root->data);preorder_recursion(root->LChild);preorder_recursion(root->RChild);
}
非递归

借助栈辅助

// 迭代前序遍历
// 注意一点:节点入栈顺序,与出栈顺序
void preorder_travel(TreeNode* root)
{assert(root);// 准备栈TreeNode* stack[1024] = { 0 };int top = -1;stack[++top] = root;while (top != -1) {TreeNode* temp = stack[top];top--;printf("%c ", temp->data);if (temp->RChild != NULL) stack[++top] = temp->RChild;if (temp->LChild != NULL) stack[++top] = temp->LChild;}
}

中序

遍历顺序:中左右

在这里插入图片描述

递归
// 递归中序
void mid_recursion(TreeNode* root)
{if (root == NULL) {printf("NULL ");return;}mid_recursion(root->LChild);printf("%c ", root->data);mid_recursion(root->RChild);
}
非递归

要注意:还有一个指针跟着栈走

// 中序,回退靠栈
// 核心:cur与栈配合,维护cur指针
void mid_travel(TreeNode* root)
{assert(root);TreeNode* stack[1024] = { 0 };int top = -1;TreeNode* cur = root;while (cur != NULL || top != -1) {if (cur != NULL) {stack[++top] = cur;cur = cur->LChild;}else {TreeNode* t = stack[top];top--;printf("%c ", t->data);cur = t->RChild;}}
}

后序

遍历顺序:左右中

在这里插入图片描述

递归
// 递归后序
void last_recursion(TreeNode* root)
{if (root == NULL) {printf("NULL ");return;}last_recursion(root->LChild);last_recursion(root->RChild);printf("%c ", root->data);
}
非递归

和前序遍历一样,维护一个栈

// 后序
// 核心:pre指针、cur指针,以及什么时候pre 与 cur指针相遇,以及为什么弹出后cur = NULL,pre = cur;
// 动画:想清楚如果一个节点要打印,则情况是什么,pre一直在模拟左、右、中节点
void last_travel(TreeNode* root)
{assert(root);TreeNode* stack[1024] = { 0 };int top = -1;TreeNode* used = NULL;TreeNode* cur = root;while (cur != NULL || top != -1) {while (cur) {stack[++top] = cur;cur = cur->LChild;}cur = stack[top];if (cur->RChild == NULL || cur->RChild == used) {printf("%c ", cur->data);top--;used = cur;cur = NULL;}else {cur = cur->RChild;}}
}

总代码

#include <stdio.h>
#include <assert.h>
#include <stdlib.h>
#include <stdbool.h>typedef struct TreeNode {char data;struct TreeNode* LChild;struct TreeNode* RChild;
}TreeNode;TreeNode* create_node(char data)
{TreeNode* new_node = (TreeNode*)calloc(1, sizeof(TreeNode));assert(new_node);new_node->data = data;return new_node;
}// 傻瓜式建立树
void create_tree(TreeNode* parent, TreeNode* lChild, TreeNode* rChild)
{// 父亲不为 NULL 即可assert(parent);parent->LChild = lChild;parent->RChild = rChild;
}// 递归前序
void preorder_recursion(TreeNode* root)
{if (root == NULL) {printf("NULL ");return;}printf("%c ", root->data);preorder_recursion(root->LChild);preorder_recursion(root->RChild);
}// 递归中序
void mid_recursion(TreeNode* root)
{if (root == NULL) {printf("NULL ");return;}mid_recursion(root->LChild);printf("%c ", root->data);mid_recursion(root->RChild);
}// 递归后序
void last_recursion(TreeNode* root)
{if (root == NULL) {printf("NULL ");return;}last_recursion(root->LChild);last_recursion(root->RChild);printf("%c ", root->data);
}// 迭代前序遍历
// 注意一点:节点入栈顺序,与出栈顺序
void preorder_travel(TreeNode* root)
{assert(root);// 准备栈TreeNode* stack[1024] = { 0 };int top = -1;stack[++top] = root;while (top != -1) {TreeNode* temp = stack[top];top--;printf("%c ", temp->data);if (temp->RChild != NULL) stack[++top] = temp->RChild;if (temp->LChild != NULL) stack[++top] = temp->LChild;}
}// 中序,回退靠栈
// 核心:cur与栈配合,维护cur指针
void mid_travel(TreeNode* root)
{assert(root);TreeNode* stack[1024] = { 0 };int top = -1;TreeNode* cur = root;while (cur != NULL || top != -1) {if (cur != NULL) {stack[++top] = cur;cur = cur->LChild;}else {TreeNode* t = stack[top];top--;printf("%c ", t->data);cur = t->RChild;}}
}// 后序
// 核心:pre指针、cur指针,以及什么时候pre 与 cur指针相遇,以及为什么弹出后cur = NULL,pre = cur;
// 动画:想清楚如果一个节点要打印,则情况是什么,pre一直在模拟左、右、中节点
void last_travel(TreeNode* root)
{assert(root);TreeNode* stack[1024] = { 0 };int top = -1;TreeNode* used = NULL;TreeNode* cur = root;while (cur != NULL || top != -1) {while (cur) {stack[++top] = cur;cur = cur->LChild;}cur = stack[top];if (cur->RChild == NULL || cur->RChild == used) {printf("%c ", cur->data);top--;used = cur;cur = NULL;}else {cur = cur->RChild;}}
}int main()
{// 一个一个创建节点TreeNode* a = create_node('a');   TreeNode* c = create_node('c');TreeNode* d = create_node('d');TreeNode* s = create_node('s');TreeNode* h = create_node('h');TreeNode* e = create_node('e');TreeNode* b = create_node('b');TreeNode* v = create_node('v');TreeNode* m = create_node('m');// 连接create_tree(a, c, d);   // a为跟节点create_tree(c, s, h);create_tree(d, e, b);create_tree(s, NULL, v);create_tree(h, NULL, NULL);create_tree(e, m, NULL);create_tree(b, NULL, NULL);printf("递归前序: \n");preorder_recursion(a);    printf("\n递归中序: \n");mid_recursion(a);printf("\n递归后序: \n");last_recursion(a);printf("\n迭代前序: \n");preorder_travel(a);printf("\n迭代中序: \n");mid_travel(a);printf("\n迭代后序: \n");last_travel(a);return 0;
}

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com