数据结构——树和二叉树

目录

一、树的概念

二、树结点之间的关系

三、二叉树

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
相关推荐
Java与Android技术栈5 分钟前
图像编辑器 Monica 之 CV 常见算法的快速调参
算法
别NULL17 分钟前
机试题——最小矩阵宽度
c++·算法·矩阵
珊瑚里的鱼18 分钟前
【单链表算法实战】解锁数据结构核心谜题——环形链表
数据结构·学习·程序人生·算法·leetcode·链表·visual studio
无限码力22 分钟前
[矩阵扩散]
数据结构·算法·华为od·笔试真题·华为od e卷真题
gentle_ice22 分钟前
leetcode——矩阵置零(java)
java·算法·leetcode·矩阵
查理零世24 分钟前
保姆级讲解 python之zip()方法实现矩阵行列转置
python·算法·矩阵
zhbi9844 分钟前
测量校准原理
算法
时间很奇妙!1 小时前
decison tree 决策树
算法·决策树·机器学习
Icomi_1 小时前
【外文原版书阅读】《机器学习前置知识》1.线性代数的重要性,初识向量以及向量加法
c语言·c++·人工智能·深度学习·神经网络·机器学习·计算机视觉
apocelipes1 小时前
Linux glibc自带哈希表的用例及性能测试
c语言·c++·哈希表·linux编程