追梦之旅【数据结构篇】——详解C语言实现二叉树

简介: 详解C语言实现二叉树~😎前言🙌什么是二叉树?二叉树的性质总结:整体实现内容分析💞1.头文件的编写:🙌2.功能文件的编写:🙌1)前序遍历的数值来创建树——递归函数实现 😊2)求树的高度函数实现 😊3)求叶子数函数实现 😊4)求树的总结点个数函数实现 😊5)前序遍历二叉树实现 😊6)中序遍历二叉树实现 😊7)后序遍历二叉树实现 😊8)删除二叉树函数实现 😊3.测试文件编写::🙌总结撒花💞

微信图片_20230427214238.gif

😎博客昵称:博客小梦

😊最喜欢的座右铭:全神贯注的上吧!!!

😊作者简介:一名热爱C/C++,算法等技术、喜爱运动、热爱K歌、敢于追梦的小博主!

😘博主小留言:哈喽!😄各位CSDN的uu们,我是你的博客好友小梦,希望我的文章可以给您带来一定的帮助,话不多说,文章推上!欢迎大家在评论区唠嗑指正,觉得好的话别忘了一键三连哦!😘


微信图片_20230427160707.gif


前言🙌



   哈喽各位友友们😊,我今天又学到了很多有趣的知识,现在迫不及待的想和大家分享一下!😘我仅已此文,手把手带领大家详解C语言实现二叉树~ 利用二叉链式存储结构来完成二叉树的实现,并完成叶子高度,前序遍历生成树,叶节点的个数,结点总数,前序遍历,中序遍历,后序遍历,销毁树。都是精华内容,可不要错过哟!!!😍😍😍


什么是二叉树?


满足以下两个条件的树就是二叉树:


1.本身是有序树;

2.树中包含的各个节点的度不能超过 2,即只能是 0、1 或者 2;


二叉树的性质总结:


1.二叉树中,第 i 层最多有 2^( i-1)个结点。

2.如果二叉树的深度为 K,那么此二叉树最多有 2^K-1 个结点。

3.二叉树中,终端结点数(叶子结点数)为 n0,度为 2 的结点数为 n2,则 n0=n2+1。


二叉树又可以分类为许多不同的二叉树:

微信图片_20230428141935.png


整体实现内容分析💞


  • 1.利用二叉链式存储结构来完成二叉树的实现,并完成叶子高度,前序遍历生成树,叶节点的个数,结点总数,前序遍历,中序遍历,后序遍历,销毁树。
  • 2.采用递归的思想,先是malloc开辟结点空间,然后给结点赋值,然后递归左子树然后递归右子树。这里用*表示空。最后返回生成的root指针的地址求高度,采用后序遍历的思想。
  • 3.相当于求左右子树结点高度的最大值。每次递归加1就是计算结点数。求叶子总数时先求出左子树的叶子数再加上右子树的叶子数。求总结点数时这里直接递归计算出左子树和右子树的总结点数,每次加1表示遍历的节点数计算。然后就是前中后序的实现,这里也是用到递归的思想,最后便是将数销毁掉,malloc生成的空间要用free手动销毁。


1.头文件的编写:🙌


头文件的编写的整体思路分析:

这里用的是二叉链式存储的实现。首先是定义结构体,然后是对求叶子高度,前序遍历生成树,叶节点的个数,结点总数,前序遍历,中序遍历,后序遍历,销毁树。

#pragma once
#include"stdio.h"
#include"stdlib.h"
typedef int datatype;
typedef struct node
{
  datatype data;
  struct node* lchild, * rchild;
}tree;
tree* Creatbitree();
int Depthbitree(tree* T);
int Leaf_count(tree* T);
int Countbitree(tree* T);
int Preorder(tree* T);
int Inorder(tree* T);
int Postorder(tree* T);
tree* Delete(tree* T);


2.功能文件的编写:🙌


1)前序遍历的数值来创建树——递归函数实现 😊


编写的整体思路分析:

采用递归的思想,先是malloc开辟结点空间,然后给结点赋值,然后递归左子树然后递归右子树。这里用*表示空。最后返回生成的root指针的地址

#include"BinaryTree.h"
tree* Creatbitree()//前序遍历的数值来创建树——递归 
{
  char ch;
  tree* root;
  scanf("%c", &ch);//用于接收输入的数值 
  if (ch == '*') return NULL;//用*来判断是否为空 
  else {
    root = (tree*)malloc(sizeof(tree));
    root->data = ch;//赋值 
    root->lchild = Creatbitree();//左子树 
    root->rchild = Creatbitree();//右子树 
  }
  return root;
}


2)求树的高度函数实现 😊


编写的整体思路分析:

这里求高度,采用后序遍历的思想。相当于求左右子树结点高度的最大值。每次递归加1 就是计算结点数。

int Depthbitree(tree* T)//测量树的深度 
{
  if (T == NULL) return 0;
  else {
    int leftheighter = Depthbitree(T->lchild);
    int rightheighter = Depthbitree(T->rchild);
    return (leftheighter > rightheighter ? leftheighter + 1 : rightheighter + 1);
  }
}


3)求叶子数函数实现 😊


编写的整体思路分析:

代码上已表明算法思想,先求出左子树的叶子数再加上右子树的叶子数。

int Leaf_count(tree* T)//测量叶子的数量 
{
  if (T == NULL) return 0;
  else if (!T->lchild && !T->rchild)//如果左右结点都为空则他就是叶子结点 
    return 1;
  else return Leaf_count(T->lchild) + Leaf_count(T->rchild);
}


4)求树的总结点个数函数实现 😊


编写的整体思路分析:

这里直接递归计算出左子树和右子树的总结点数,每次加1表示遍历的节点数计算。

int Countbitree(tree* T) //测量总的结点个数
{
  if (T == NULL)
    return 0;
  else {
    return  Countbitree(T->lchild) + Countbitree(T->rchild)+1;
  }
}


5)前序遍历二叉树实现 😊


int Preorder(tree* T)//前序遍历序列 (根左右) 
{
  if (T == NULL)
    return 0;
  else {
    printf("%c  ", T->data);//先输出根节点 
    Preorder(T->lchild);
    Preorder(T->rchild);
  }
}


6)中序遍历二叉树实现 😊


int Inorder(tree* T)//中序遍历序列 (左根右) 
{
  if (T == NULL)
    return 0;
  else {
    Inorder(T->lchild);//先输出左孩子 
    printf("%c  ", T->data);
    Inorder(T->rchild);
  }
}


7)后序遍历二叉树实现 😊


int Postorder(tree* T)//后序遍历序列 (左右根) 
{
  if (T == NULL)
    return 0;
  else {
    Postorder(T->lchild);//先输出左孩子 
    Postorder(T->rchild);
    printf("%c  ", T->data);
  }
}


8)删除二叉树函数实现 😊


tree* Delete(tree* T)//删除树 
{
  if (T->lchild)
    Delete(T->lchild);
  else if (T->rchild)
    Delete(T->rchild);
  else
    free(T);
}


3.测试文件编写::🙌


#define _CRT_SECURE_NO_WARNINGS 1
#include"BinaryTree.h"
main()
{
  tree* T;
  T = (tree*)malloc(sizeof(tree));
  T->lchild = NULL;
  T->rchild = NULL;
  printf("请输入树的前序遍历序列\n");
  T = Creatbitree();
  int n = Depthbitree(T);
  int m = Leaf_count(T);
  int l = Countbitree(T);
  printf("树创建完成\n");
  printf("前序输出为\n");
  printf("\t\t");
  Preorder(T);
  printf("\n中序输出为\n");
  printf("\t\t");
  Inorder(T);
  printf("\n后序输出为\n");
  printf("\t\t");
  Postorder(T);
  printf("\n高度%d\n叶子%d\n总结点%d\n", n, m, l);
  Delete(T);
  printf("删除成功");
}


功能测试结果展示图:

微信图片_20230428142819.png


总结撒花💞


本篇文章旨在分享详解C语言实现二叉树。希望大家通过阅读此文有所收获!本次主要是对二叉树的实现,这里只要用到的思想是递归,这也是难点所在。这就要需要画图帮忙辅助理解,递归的具体每一步是如何执行的需要分析清楚。在创建树的时候可以采用前序遍历思想创建,这种思想创建是比较好理解的,也可以用其他思想创建,相对比较难理解一点。以及区分好前中后序遍历的思想,然后再编写代码。

  😘如果我写的有什么不好之处,请在文章下方给出你宝贵的意见😊。如果觉得我写的好的话请点个赞赞和关注哦~😘😘😘


相关文章
|
22天前
|
C语言
【数据结构】栈和队列(c语言实现)(附源码)
本文介绍了栈和队列两种数据结构。栈是一种只能在一端进行插入和删除操作的线性表,遵循“先进后出”原则;队列则在一端插入、另一端删除,遵循“先进先出”原则。文章详细讲解了栈和队列的结构定义、方法声明及实现,并提供了完整的代码示例。栈和队列在实际应用中非常广泛,如二叉树的层序遍历和快速排序的非递归实现等。
109 9
|
21天前
|
存储 搜索推荐 算法
【数据结构】树型结构详解 + 堆的实现(c语言)(附源码)
本文介绍了树和二叉树的基本概念及结构,重点讲解了堆这一重要的数据结构。堆是一种特殊的完全二叉树,常用于实现优先队列和高效的排序算法(如堆排序)。文章详细描述了堆的性质、存储方式及其实现方法,包括插入、删除和取堆顶数据等操作的具体实现。通过这些内容,读者可以全面了解堆的原理和应用。
60 16
|
21天前
|
C语言
【数据结构】二叉树(c语言)(附源码)
本文介绍了如何使用链式结构实现二叉树的基本功能,包括前序、中序、后序和层序遍历,统计节点个数和树的高度,查找节点,判断是否为完全二叉树,以及销毁二叉树。通过手动创建一棵二叉树,详细讲解了每个功能的实现方法和代码示例,帮助读者深入理解递归和数据结构的应用。
72 8
|
24天前
|
存储 C语言
【数据结构】手把手教你单链表(c语言)(附源码)
本文介绍了单链表的基本概念、结构定义及其实现方法。单链表是一种内存地址不连续但逻辑顺序连续的数据结构,每个节点包含数据域和指针域。文章详细讲解了单链表的常见操作,如头插、尾插、头删、尾删、查找、指定位置插入和删除等,并提供了完整的C语言代码示例。通过学习单链表,可以更好地理解数据结构的底层逻辑,提高编程能力。
50 4
|
25天前
|
存储 C语言
【数据结构】顺序表(c语言实现)(附源码)
本文介绍了线性表和顺序表的基本概念及其实现。线性表是一种有限序列,常见的线性表有顺序表、链表、栈、队列等。顺序表是一种基于连续内存地址存储数据的数据结构,其底层逻辑是数组。文章详细讲解了静态顺序表和动态顺序表的区别,并重点介绍了动态顺序表的实现,包括初始化、销毁、打印、增删查改等操作。最后,文章总结了顺序表的时间复杂度和局限性,并预告了后续关于链表的内容。
56 3
|
24天前
|
C语言
【数据结构】双向带头循环链表(c语言)(附源码)
本文介绍了双向带头循环链表的概念和实现。双向带头循环链表具有三个关键点:双向、带头和循环。与单链表相比,它的头插、尾插、头删、尾删等操作的时间复杂度均为O(1),提高了运行效率。文章详细讲解了链表的结构定义、方法声明和实现,包括创建新节点、初始化、打印、判断是否为空、插入和删除节点等操作。最后提供了完整的代码示例。
40 0
|
1月前
|
C语言 C++
C语言 之 内存函数
C语言 之 内存函数
35 3
|
14天前
|
C语言
c语言调用的函数的声明
被调用的函数的声明: 一个函数调用另一个函数需具备的条件: 首先被调用的函数必须是已经存在的函数,即头文件中存在或已经定义过; 如果使用库函数,一般应该在本文件开头用#include命令将调用有关库函数时在所需要用到的信息“包含”到本文件中。.h文件是头文件所用的后缀。 如果使用用户自己定义的函数,而且该函数与使用它的函数在同一个文件中,一般还应该在主调函数中对被调用的函数做声明。 如果被调用的函数定义出现在主调函数之前可以不必声明。 如果已在所有函数定义之前,在函数的外部已做了函数声明,则在各个主调函数中不必多所调用的函数在做声明
29 6
|
1月前
|
存储 缓存 C语言
【c语言】简单的算术操作符、输入输出函数
本文介绍了C语言中的算术操作符、赋值操作符、单目操作符以及输入输出函数 `printf` 和 `scanf` 的基本用法。算术操作符包括加、减、乘、除和求余,其中除法和求余运算有特殊规则。赋值操作符用于给变量赋值,并支持复合赋值。单目操作符包括自增自减、正负号和强制类型转换。输入输出函数 `printf` 和 `scanf` 用于格式化输入和输出,支持多种占位符和格式控制。通过示例代码详细解释了这些操作符和函数的使用方法。
39 10
|
27天前
|
存储 算法 程序员
C语言:库函数
C语言的库函数是预定义的函数,用于执行常见的编程任务,如输入输出、字符串处理、数学运算等。使用库函数可以简化编程工作,提高开发效率。C标准库提供了丰富的函数,满足各种需求。