C语言 数据结构 树和二叉树

1、树:是n节点的有限集。树是n(n=>0)个节点的有限集。 n=0时成为空树。

在任意一颗非空树中:(1)有且仅有一个称为根的节点;(2)当n>0时,其余节点可分为m(m>0)个互不相交的有限集T1、T2、T3、Tm,其中每个节点又是一棵树,并且称为根的子树。

2 、节点分类:节点拥有的的子树数称为节点的度。度为0的称为终端节点或者叶节点;度不为0的称为非终端节点或者分支节点树的度是树内节点的度的最大值。

3、节点间关系:节点的子树的根称为该节点的孩子, 该节点称为孩子的双亲。同一个双亲的孩子之间称为兄弟。如果将树中节点的各子树看成是从左到右是有次序的,不能互换的,则称该树为有序树,否则称为无序树。

4、结点的层次:从根开始起根为第一层,根的孩子为第二层,树中结点最大层次成为树的深度或高度。如果树中结点的各子树看做是有序的不能互换的则称该树为有序树,否则为无序树。森林是m(m>=0)颗互不相交的树的集合。

5、线性结构和树结构对比:线性结构的第一个元素无前驱,树结构的根节点无双亲,且唯一。线性表有一个元素无后继, 树节点的叶节点无孩子,但可以有多个,线性结构中间元素一个前驱一个后继,树节点一个双亲多个孩子。

树的抽象数据类型定义:

6、树的存储结构:

存储结构:顺序存储、链式存储

问题:用一段连续的存储空间(数组)无论如何设计也无法反映树的逻辑关系。谁是谁的父母?谁是谁的孩子?谁是谁的兄弟?。。。因此,简单的顺序存储无法表示树的逻辑关系。

解决办法:利用顺序结构结合链式结构的特点!!!

设计思想:双亲表示法、孩子表示法、孩子兄弟表示法。

双亲表示法:以一组连续空间存储树的结点,同时每个结点中附设一个指示器指示其双亲结点在数组中的位置。特点:找根容易O(1)找孩子难。

 

#define MAX_TREE_SIZE100;

typedef int TElemType;

typedef struct PTNode{

   TElemType data;

   int parent;

}PTNode;

typedef struct{

   PTNode nodes{MAX_TREE_SIZE];

   int r,n;  //根的位置,节点数

}PTree;

找孩子难怎么办?。。。增加长子域。如果是2个以上孩子,还得有次子域。。。3个呢。。。

如果节点的孩子数目很多,超过了2个,我们又要关注节点的双亲,又要关注节点的孩子,还要关注节点的兄弟,而且对时间遍历要求还比较高,那么我们就可以把此结构拓展设计为有双亲域、孩子域,兄弟域的结构。因此存储结构的设计是一个非常灵活的过程,一个存储结构设计是否合理,取决于数据结构的运算是否简单、方便、时间空间复杂度低。一定要注意原则:适合就好。不是越复杂越好。

 

孩子表示法思路:把每个节点的孩子节点排列起来,以单链表做存储结构,则n个节点有n个孩子链表,如果是叶子节点则单链表为空,然后n个头指针又组成一个线性表,采用顺序存储结构,存放进一个一位数组中。

找双亲不容易怎么破?增加双亲域,变成双亲孩子表示法

双亲孩子表示法:是孩子表示法的改进,添加了双亲的存储结构

孩子兄弟表示法:任意一棵树,它的节点的第一个孩子如果存在就是唯一的,它的右兄弟如果存在也是唯一的。因此我们设置两个指针分别指向该节点的第一个孩子和此节点的右兄弟。节点结构

data

firstchild

rightsib



孩子兄弟表示法意义重大:把一颗复杂的树变成了二叉树

二叉树定义二叉树(Binary Tree)是n(n>=0)个节点的有限集合,该集合或者为空集(称为空二叉树),或者由一个根节点和两颗互不相交的、分别称为根节点的左子树、右子树的二叉树组成。五种基本形态:空二叉树、只有一个根节点、根节点只有左子树、根节点只有右子树、根节点既有左子树又有右子树。

二叉树的特点

1.每个节点最多有两棵子树,即不存在度大于2的节点。

2.左子树和右子树是有序的,不能随便颠倒。

3.即使树中某个节点只有一棵子树,也要区分是左子树还是右子树。

满二叉树:

深度为k且含有2k-1个节点的二叉树成为满二叉树。

完全二叉树:

如果一个有n个节点的二叉树按满二叉树方式自上而下,自左至右对它进行编号,若树中所有节点和满二叉树的1~n编号完全一致,则称该树为完全二叉树。

错误示例:

二叉树性质:

1.在二叉树的第i层上最多有2 i-1个节点 。(i>=1)。

2.二叉树中如果深度为k,那么最多有2k-1个节点。(k>=1)。

3.n0=n2+1  n0表示度数为0的节点(叶子节点) n2表示度数为2的节点。

4.在完全二叉树中,具有n个节点的完全二叉树的深度为[log2n]+1,其中[log2n]+1是向下取整(不大于[log2n]+1的最大整数)。

5.若对含 n 个结点的完全二叉树从上到下且从左至右进行 1 至 n 的编号,则对完全二叉树中任意一个编号为 i 的结点:

(1)若 i=1,则该结点是二叉树的根,无双亲, 否则(i>1时),编号为[i/2] (向下取整)的结点为其双亲结点; 

(2)若 2i>n,则该结点无左孩子,  否则,编号为 2i 的结点为其左孩子结点;

(3)若 2i+1>n,则该结点无右孩子结点,  否则,编号为2i+1 的结点为其右孩子结点。

7、二叉树的存储

前面分析已知,树用顺序存储结构实现是比较困难的,但是二叉树是一种特殊的树,因此完全可以用顺序存储结构来描述其关系。

其双亲、孩子、兄弟的关系可以用二叉树的性质分析得到!!!!!!

顺序存储结构一般只用于完全二叉树,否则会造成空间浪费。

二叉树的链式存储结构:二叉链表(左右孩子),三叉链表(左右孩子带双亲),线索链表。

二叉树遍历:从根结点出发,按照某种次序依次访问二叉树中的节点,使得每个节点被访问一次,而且仅被访问一次。

遍历次序:前提是限制从左到右:前序遍历(根左右)、中序遍历(左根右)、后序遍历(左右根)、层序遍历。

前序遍历(根左右)ABDGHCEIF

中序遍历(左根右)GDHBAEICF

 

 

 

后序遍历(左右根)GHDBIEFCA

层序遍历 ABCDEFGHI

 

二叉树遍历(递归算法):

面试题目:推导遍历结果。《大话数据结构P212》

二叉树建立:

前序:AB#D##C##

中序:#B#D#A#C#

后序:###DB##CA

8、线索二叉树指向前驱或者后继得指针称为线索,加上线索的二叉树链表称为线性链表,相应的二叉树就称为线索二叉树。(n个结点的二叉链表一共2n个指针域,一共n-1条分支数一共2n-(n-1)=n+1个空指针域。)

9、树、森林、二叉树的相互转化

将树转换成二叉树的步骤是:
(1)加线。就是在所有兄弟结点之间加一条连线;
(2)抹线。就是对树中的每个结点,只保留他与第一个孩子结点之间的连线,删除它与其它孩子结点之间的连线;
(3)旋转。以树的根结点为轴心,将整棵树顺时针旋转一定角度,使之结构层次分明。

注意事项:第一个孩子是二叉树结点的左孩子,兄弟转换过来的孩子是右孩子!!!

比如FG是右孩子,J是右孩子,而H、I是左孩子

森林转换为二叉树:森林是由若干棵树组成,可以将森林中的每棵树的根结点看作是兄弟,由于每棵树都可以转换为二叉树,所以森林也可以转换为二叉树。将森林转换为二叉树的步骤是:
(1)先把每棵树转换为二叉树;
(2)第一棵二叉树不动,从第二棵二叉树开始,依次把后一棵二叉树的根结点作为前一棵二叉树的根结点的右孩子结点,用线连接起来。当所有的二叉树连接起来后得到的二叉树就是由森林转换得到的二叉树。

二叉树转换为树:二叉树转换为树是树转换为二叉树的逆过程,其步骤是:
(1)若某结点的左孩子结点存在,将左孩子结点的右孩子结点、右孩子结点的右孩子结点……都作为该结点的孩子结点,将该结点与这些右孩子结点用线连接起来;
(2)删除原二叉树中所有结点与其右孩子结点的连线;(3)整理(1)和(2)两步得到的树,使之结构层次分明。

二叉树转换为森林:二叉树转换为森林比较简单,其步骤如下:
(1)先把每个结点与右孩子结点的连线删除,得到分离的二叉树;
(2)把分离后的每棵二叉树转换为树;(3)整理第(2)步得到的树,使之规范,这样得到森林。

根据树与二叉树的转换关系以及二叉树的遍历定义可以推知,树的先序遍历与其转换的相应的二叉树的先序遍历的结果序列相同;树的后序遍历与其转换的二叉树的中序遍历的结果序列相同;树的层序遍历与其转换的二叉树的后序遍历的结果序列相同。由森林与二叉树的转换关系以及森林与二叉树的遍历定义可知,森林的先序遍历和中序遍历与所转换得到的二叉树的先序遍历和中序遍历的结果序列相同。

10、哈夫曼树我们称判定过程最优的二叉树为哈夫曼树,又称最优二叉树。

路径:树中一个结点到另一个结点之间的分支构成这两个结点之间的路径。

路径长度:路径上的分枝数目称作路径长度。

树的路径长度:从树根到每一个结点的路径长度之和。
结点的带权路径长度:在一棵树中,如果其结点上附带有一个权值,通常把该结点的路径长度与该结点上的权值之积称为该结点的带权路径长度。

树的带权路径长度:如果树中每个叶子上都带有一个权值,则把树中所有叶子的带权路径长度之和称为树的带权路径长度。其中带权路径长度最小的二叉树就称为哈夫曼树或最优二叉树。

哈夫曼树的构造:

按照构造哈夫曼树得到的编码成为哈夫曼编码。常用于压缩。

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/463446.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Oracle开启关闭归档日志

开启归档日志 shutdown immediate; --关闭数据库 startup mount; --打开数据库 alter database archivelog; --开启归档日志 alter database open; --开启数据库 archive log list; --查看归档日志是否开启 关闭归档日志 shutdown immediate; --关闭数据库 startup mount; …

Redis源码分析之anet网络通信的封装

anet是redis对tcp/ip网络中socket api接口的一个全面的封装,针对server/client端。封装的api的接口如下,注释了主要的接口: // tcp连接 int anetTcpConnect(char *err, char *addr, int port); // 非阻塞连接 int anetTcpNonBlockConnect(ch…

使用AJAX Toolkit创建新闻列表

我们很多站点上面都需要显示新闻列表,由标题和正文组成的。一般客户都希望实现这样的效果: 开始的时候只是显示标题,当点击标题的时候,再展开正文。再点击,又可缩回去。 这是典型的AJAX效果,或者说以前你也…

C语言 嵌入式 面试小知识点(一)

sizeof是C/C中的一个操作符(operator),简单的说其作用就是返回一个对象或者类型所占的内存字节数。 与strlen的区别: 一、sizeof 是运算符,确切的说是一个编译时运算符,参数可以是数组、指针、类型、对象…

将数据渲染到页面的几种方式

将数据渲染到页面的几种方式: 1.字符串拼接; 2.dom回流 3.文档碎片(文档碎片节点:documentFragment) 4.模板 (下章会详细介绍模板)转载于:https://www.cnblogs.com/shangjun6/p/1039740…

论证是一门学问

本文的标题借用了安东尼.韦斯顿(Anthony Weston)的《论证是一门学问》一书的标题,向安东尼老爷子致敬的同时,也希望更多人能够真正了解“什么是论证”。 争论与论证从来都不是新鲜事物,作为软件行业的科技工作者&…

[翻译]SQL Server 工作集消息

Q:我发现有指向工作集(SQL Server保留内存区域)被分页出来相关的消息: 重要部分的 SQL 服务器进程内存已被分页。这可能导致性能下降。持续时间: 0 秒。 工作集 (KB): 2484,已提交 (KB): 48036&…

Redis源码分析之工具类util

在redis源码中的辅助工具类中,主要包括大小端转换、SHA算法以及util.h中对应的算法。 大小端转换: LittleEndian:低位字节数据存放于低地址,高位字节数据存放于高地址。 BigEndian:低位字节数据存放于高地址&#x…

Linux下如何安装软件

一、解析Linux应用软件安装包通常Linux应用软件的安装包有三种:1) tar包,如software-1.2.3-1.tar.gz。它是使用UNIX系统的打包工具tar打包的。2) rpm包,如software-1.2.3-1.i386.rpm。它是RedHat Linux提供的一种包封装…

Python深浅拷贝辨析

1 import copy2 3 list1 [11, 22, [33, 44]]4 list2 list15 list3 list1[:]6 list4 copy.copy(list1)7 list5 copy.deepcopy(list1)8 9 list1[0] 0 # 对列表的首层做增删改查操作 10 print("list1:",id(list1),list1) # list1: 1455502266696 [0, 22, […

生活规则

1.朋友请你吃饭,不要觉得理所当然,请礼尚往来,否则你的名声会越来越臭。 2.给自己定目标,一年,两年,五年,也许你出生不如别人好,通过努力,往往可以改变70%的命运。破罐子…

[AX]AX2012 SSRS报表使用Report Data Method

在AX2012的SSRS报表中可以使用c#或者Visual basic .net编写report data method来获取和操作数据,由report data method返回的数据可以用在报表的表达式中,也可以用作dataset的数据源。 使用Report data method首先需要创建AX model工程,在工程…

HIVE和HBASE区别

转载:https://www.cnblogs.com/justinzhang/p/4273470.html 1. 两者分别是什么? Apache Hive是一个构建在Hadoop基础设施之上的数据仓库。通过Hive可以使用HQL语言查询存放在HDFS上的数据。HQL是一种类SQL语言,这种语言最终被转化为Map/Re…

php调试

今天在使用php 的session 的时候,出现了以前就遇见但是又解决不了的问题,在页面上出现如下提示: Warning: session_start() [function.session-start]: Cannot send session cache limiter - headers already sent (output started at E:\php…

C 结构体

C 结构体C 数组允许定义可存储相同类型数据项的变量,结构是 C 编程中另一种用户自定义的可用的数据类型,它允许您存储不同类型的数据项。结构用于表示一条记录,假设您想要跟踪图书馆中书本的动态,您可能需要跟踪每本书的下列属性&…

lightoj1259 线性筛的另一种写法 v变成bool标记数组

也是用线性筛&#xff0c;但是v用int会爆&#xff0c;所以这个线性筛用的是另外一种写法 #include<cstdio> #include<cmath> #include<queue> #include<vector> #include<cstring> #include<iostream> #include<algorithm> using na…

Redis基数统计之HyperLogLog小内存大用处

转载&#xff1a;https://blog.csdn.net/azhegps/article/details/71158952 我们一直都知道&#xff0c;redis几大常用数据结构&#xff0c;字符串、散列、列表、集合、有序集合。其实后来Redis做了很多补充&#xff0c;其中之一就是HyperLogLog&#xff0c;另外的还有GEO&…

matlab中的qr函数

转自&#xff1a;https://blog.csdn.net/qq278672818/article/details/62038630 实数矩阵A的QR分解是把A分解为A QR这里的Q是正交矩阵&#xff08;意味着QTQ I&#xff09;而R是上三角矩阵。类似的&#xff0c;我们可以定义A的QL, RQ和LQ分解。更一般的说&#xff0c;我们可以…

(String) 和 String.valueOf() 两种字符串转换的区别

使用 String.valueOf() 进行数据转换&#xff0c;如果被转换的数据为 null, 则这种方法将返回一个 "null" 字符串 &#xff08;String&#xff09; 方法进行转换时&#xff0c;如果被转换的数据为 null, 则返回 null 对象而不是一个 "null" 字符串。转载于…

利用有名管道实现进程间的通信

1 /*****************************************************************2 * Copyright (C) 2018 FBI WARNING. All rights reserved.3 * 4 * 文件名称&#xff1a;fifo_write.c5 * 创 建 者&#xff1a;constantine6 * 创建日期&#xff1a;2018年02月26日7 * 描 …