数据结构——树和二叉树

目录

一、树的概念

二、树结点之间的关系

三、二叉树

1、满二叉树

2、完全二叉树

四、二叉树的存储

1、顺序存储

2、链式存储

一、树的概念

如果数据和数据之间满足一对多的关系,将其逻辑结构称之为树

如下图:树的根与树的分支存在一对多的关系

将上图具体化,从根节点开始,每个节点都有一个或者多个分支,当然还有没有分支的情况

  1. 如果数据和数据之间满足一对多的关系,将其逻辑结构称之为树。
    1. 逻辑关系:树形关系
    2. 存储关系:顺序存储 链式存储
  2. 每棵子树的根节点有且仅有一个直接前驱,但是可以有0个或者多个后继。
    1. 子树:一个结点及其下面的所有结点组成的树。

二、树结点之间的关系

  1. 根节点:最上层的第一个结点,每棵树有且仅有一个根节点。根节点没有前驱
  2. 父结点:上一层与自己紧挨着的结点,称之为双亲结点。
  3. 子结点:下一层与自己紧挨着的结点。
  4. 兄弟结点:
  5. 堂兄弟结点:
  6. 叶子结点:没有子结点的结点

  1. 度:
    1. 结点的度:该结点的子结点个数。
    2. 树的度:该树中节点的最大的度
  2. 层数:
    1. 根结点的层数是1
    2. 结点的层数:父结点层数+1
  3. 深度:
    1. 从根结点出发,到最底层的层数

举个例子:

上图当中:A称为根,BCD是A的子结点,称A是BCD的父结点,B,C,D之间是兄弟结点,E和G是堂兄弟结点,这个关系可以参考家庭之间的关系

叶子结点(终端结点):比如E,F没有后继结点的结点称为叶子结点

分支结点(非叶子结点):有后继结点,比如G结点

A的度:3 B的度:2 D的度:4 上图树的度:4 上图树的层数:4

A所在层数:1 C所在层数:2

4.有序树与无序树

1.如果将树中结点的各个子树看成是从左往右有次序的(即不能交换),则称该树为有序树,否则称为无序树

森林:多个树组成的一个集合,树之间不存在关系,每棵树都独立存在

如下图三个树的集合:

三、二叉树

二叉树是树的一种特殊形式,即每个分支最多只有两个子结点

1、满二叉树

每个结点的度都为2,即每个结点都有两个子节点,(即二叉树中每个结点的子结点都是满的)

2、完全二叉树

在满二叉树的最底层,最右边,删除n个连续结点,形成的二叉树称之为完全二叉树。

完全二叉树的特点:

  1. 叶子结点只能出现在最下层和次下层
  2. 最下层的叶子结点集中在树的左边
  3. 倒数第2层的叶子结点,一定在右边连续位置
  4. 满二叉树一定是完全二叉树,完全二叉树不一定是满二叉树

几种常见的完全二叉树:

完全二叉树的编号方式:按照顺序编号

四、二叉树的存储

1、顺序存储

  1. 二叉树的顺序存储,就是使用一个一维数组存储二叉树结点中的数据,且结点的编号位置就是数组的下标索引。
  2. 只有在完全二叉树的情况下才能填满数组
  3. 当为不完全二叉树的时候,会造成大量的空间浪费。所以一般不使用顺序存储。

2、链式存储

  1. 先序,中序,后序创建。

  2. 由于是单向链表前者结点可以找到后者结点,但是后者结点不好找前者结点。

  3. 所以使用先序创建。

  4. 结点的表示:

    包含了数据域、指向左子树的指针和指向右子树的指针

    数据域:存放的数据

    指向左子树的指针:存放左子树的地址

    指向右子树的指针:存放右子树的地址

使用链式存储方式实现二叉树的创建和遍历,创建如下图的二叉树

cs 复制代码
//二叉树的应用
#include<stdio.h>
#include<stdlib.h>
#include<stdbool.h>
//创建根节点
struct tree_node
{
	char date;
	struct tree_node *left;//左树指针
	struct tree_node *right;//右树指针
};
//先序创建
//利用递归创建二叉树,返回类型为下一个结点的地址,实现递归创建
struct tree_node * tree_create()
{
	struct tree_node *node=NULL;//创建一个指针存放节点地址,初始化为空
	char str;//用来存储创建的数据
	scanf(" %c",&str);//从终端获取要创建的二叉树
	if(str=='#')//如果接受到字符#,即这个结点没有
		return NULL;
	node=(struct tree_node*)malloc(sizeof(struct tree_node));//新创建的结点
	if(node==NULL)
	{
		printf("结点创建失败\n");
		return NULL;
	}
	node->date=str;
	//递归创建左子树
	node->left=tree_create();
	//递归创建右子树
	node->right=tree_create();
	return node;
}
//先序遍历
bool tree_erg_fir(struct tree_node *root)
{
	if(root==NULL)
	{
		return false;
	}
	printf("%c",root->date);
	tree_erg_fir(root->left);
	tree_erg_fir(root->right);
}
//中序遍历
bool tree_erg_middle(struct tree_node *root)
{
	if(root==NULL)
	{
		return false;
	}
	tree_erg_middle(root->left);
	printf("%c",root->date);
	tree_erg_middle(root->right);
}
//后序遍历
bool tree_erg_behind(struct tree_node *root)
{
	if(root==NULL)
	{
		return false;
	}
	tree_erg_behind(root->left);
	tree_erg_behind(root->right);
	printf("%c",root->date);
}
int main(int argc, const char *argv[])
{
	printf("请输入二叉树,创建方式为先序创建:\n");
	struct tree_node *root=tree_create();
	//先序遍历
	tree_erg_fir(root);
	printf("\n");
	//中序遍历
	tree_erg_middle(root);
	printf("\n");
	//后序遍历
	tree_erg_behind(root);
	printf("\n");

	return 0;
}

输出如下;

cs 复制代码
请输入二叉树,创建方式为先序创建:
ABDE###F##CM###
ABDEFCM
EDBFAMC
EDFBMCA
相关推荐
jianglq9 分钟前
C++20 协程:异步编程的新纪元
算法·c++20
Ajiang282473530440 分钟前
贪吃蛇项目实现(C语言)——附源码
c语言·开发语言
wniuniu_1 小时前
动态规划前---选----
算法·动态规划
lzb_kkk2 小时前
【Redis】redis5种数据类型(哈希)
开发语言·redis·算法·缓存·哈希算法
鱼跃鹰飞2 小时前
Leetcode面试经典150题-202.快乐数
算法·leetcode·面试
鱼跃鹰飞2 小时前
Leetcode面试经典150题-82.删除排序链表中的重复元素II
算法·leetcode·面试
格林威3 小时前
Baumer工业相机堡盟工业相机如何通过BGAPISDK使用短曝光功能(曝光可设置1微秒)(C语言)
c语言·开发语言·人工智能·数码相机·计算机视觉
结衣结衣.4 小时前
Linux——进程状态
linux·运维·服务器·c语言·笔记·学习
云边有个稻草人4 小时前
【刷题】Day4--密码检查
开发语言·数据结构·笔记·算法
格林威4 小时前
Baumer工业相机堡盟工业相机如何通过BGAPI SDK设置相机的图像剪切(ROI)功能(C语言)
c语言·开发语言·人工智能·数码相机·计算机视觉