数据结构——树和二叉树

目录

一、树的概念

 二、树结点之间的关系

三、二叉树

 1、满二叉树

2、完全二叉树

四、二叉树的存储

 1、顺序存储

2、链式存储

 

 

 

 

一、树的概念

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

 

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

b54e4257f7104888806cc26e65f1dcf5.jpeg

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

         6da1bbb1c5a94f73a8d5d6383acfeee5.png

 

 

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

 

 二、树结点之间的关系

 

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

-------------------------------------

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

 举个例子:

6da1bbb1c5a94f73a8d5d6383acfeee5.png

    

上图当中: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.如果将树中结点的各个子树看成是从左往右有次序的(即不能交换),则称该树为有序树,否则称为无序树

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

        如下图三个树的集合:

21542c475c3d4ffa8550d575a19891d0.png

 

三、二叉树

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

        55c09fa69acd4803b253821ac58017c3.png 

 1、满二叉树

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

edf5e215872843c1be3e1d835c4f1541.png

 

2、完全二叉树

        

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

完全二叉树的特点:

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

        几种常见的完全二叉树:

8178040ca33746a19d93f77dfaaa6b45.png

       

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

b55cd3300cbc4b99a0bcad554f3adefb.png

 

四、二叉树的存储

 1、顺序存储

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

2、链式存储

  1. 先序,中序,后序创建。
  2. 由于是单向链表前者结点可以找到后者结点,但是后者结点不好找前者结点。
  3. 所以使用先序创建。
  4. 结点的表示:

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

    数据域:存放的数据

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

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

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

023ff6c0a31e4ead9919e83597d99bca.png

//二叉树的应用
#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;
}

输出如下;

请输入二叉树,创建方式为先序创建:
ABDE###F##CM###
ABDEFCM
EDBFAMC
EDFBMCA

 

 

  • 8
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值