数据结构笔记二叉树与树

二叉树的一些概念:

//二叉树有严格的左右子树之分,度数为2的树则没有对此进行要求。

//有序树:如果将树中结点的各子树看成从左至右是有次序的,不能互换的,则称该树为有序树,否则称为无序树。

//度数为2的有序树不能称为二叉树,因为当某个结点度数为1的时候,没有左右之分。

二叉树的性质:

  1. 非空二叉树上的叶结点等于度为 2的结点数+1

//也是由叶子结点+非叶子结点= n = 孩子结点 + 1推导出;

孩子结点:结点的度数为1度数为2度数为0的情况之和(如果是完全二叉树就没有度数为1的情况,就直接是2 * m(m是非叶子结点的个数)),相当是没有根结点的层序遍历。

设度数为012的结点个数分别为n1, n2, n3

则有n1+n2+n3 = n;

由上面孩子结点知:n = n2*1 + n3*2 + 1;//不算度数为0得情况,因为度数为0是叶子结点没有孩子

两式联立得,n1 = n3 + 1;即叶结点等于度为 2的结点数+1

常见的两种二叉树:

//如上E为满二叉树,每一层的结点个数都达到了当层能达到的最大结点数

//满二叉树自上而下,从左到右依次进行编号,当且仅当二叉树中的每一个结点的编号都与该满二叉树的结点的编号一一对应,称之为完全二叉树

//理论上可以使用数组来存储完全二叉树,用数组下标作为结点编号,则该结点的左子树就为2*x, 2*x+1,可以找到它的左右结点

//不能通过该结点的编号来进行判断是否为叶子结点,因为只有最右边的结点的编号等于结点个数,最后一层的其他结点的编号都是小于结点数的,因此无法判断,一般通过判断其左子树的编号是否大于结点个数即可,为什么不用判断右子树,因为没有必要,如下图编号为4的结点,左子树的编号即为8,大于7,最小的编号都大于了,右子树肯定也大于了,没必要判断

//换句话来说,也就是int id, if(id > n) return ;当当前结点的编号大于元素个数,则说明达到了空结点

完全二叉树:

①完全二叉树的定义:完全二叉树是指完全填充的树,但底层可能除外,底层是从左到右填充的。

②完全二叉树的存储除了用二叉链表存储以外还可以用更方便的数组进行存储,从1开始(根节点下标,1号位存放的必须是根节点),到n,下标为其结点的编号,值为权重。

这样就有:完全二叉树中任意一个结点编号为x,则左节点一定是2x,右节点的编号一定是2x+1

除以以外,该数组中元素的存放顺序恰好是该完全二叉树的层序遍历序列。

//一定要记住,二叉树有0,1,2这三种度,结点个数= 这三种度数结点个数之和,千万不要忘记度数为1的情况,也就是完全二叉树和二叉树一定要区分开!!!

对二叉树进行操作

二叉树的查找、修改、插入与删除

//

//修改指针就需要加引用修改指针本身,如果是修改指针所指向的内容就没必要,按值传参使用指针的副本即可。

//

//

二叉树的遍历

(11条消息) 二叉树的性质和遍历_柘木木的博客-CSDN博客

//遍历顺序中,左子树一定先于右子树,所谓“先中后”苏配置的是跟接待你在遍历中的位置

先序遍历:(根左右)

//得到的先序遍历的序列为ABDECF

//由图我们可以知道A的子树有哪些结点,右子树有哪些结点,如果只给出一个序列,则我们不能知道左子树的序列多有长,也就不知道右子树的根节点是多少,因此不能确定一棵树

中序遍历:(左根右)

//同样使用该图,得到的中序遍历DBEACF

//通过中序遍历序列区分左右子树序列,通过先序遍历或者后序遍历序列或者层序遍历序列确定根节点

后序遍历:(左右根)

//后序遍历得到的序列为:DEBFCA

//利用层序遍历找到根节点,中序遍历分出左右子树,知道左右子树序列长度放到层序遍历序列中又能找出,左右子树的根节点,这样子一个个根节点找出来,从上到下确定一颗二叉树。

//要保证所有元素不相同,否则可能会找错根节点从而分错左右子树

层序遍历:(按层次的顺序,从上到下,从左到右)

层次遍历就相当于从根结点开始广度优先搜索

基本思路如下:

//定制队列的时候是queue<node* > q而不是 queue<node >q;

//入队的时候,孩子结点的层次是父亲结点的层次+1即可,记得根节点的层次是1;

最后一个关于二叉树遍历的重要问题

//如先序遍历,根据中序遍历序列得出左右子树的长度,A/BC/DEF,B和D又是一个根结点,且为A的左右孩子,即一次遍历能确定两个左右孩子,从而确定出一颗二叉树,如下图

//最后一次序列的长度应该是1,即该子树只有一个根节点没有左孩子右孩子,再往下递归先序序列的长度就小于或者等于0了

//建树的过程有点像先序遍历(根左右);

遍历二叉树需要掌握:

1.如何进行先序,中序,后序遍历一颗二叉树

2.如何利用中序遍历序列和任何一个遍历序列去确定一棵二叉树

3.如何反转一颗二叉树

树的遍历:

树的遍历的方式只有两种,分别是先根遍历和层序遍历

(11条消息) 笔记-树的遍历_柘木木的博客-CSDN博客//代码和总结

  1. 先根遍历 : 总是先访问根结点,再去访问所有子树

  1. 和二叉树层序遍历的思想是一样的,总是从树根开始,自上到下,从左到右一层层地向下遍历

树的遍历和DFS和BFS的关系

//树的先根遍历就是DFS,深度优先搜索以"深度"作为第一关键词进行对结点遍历

//“剪枝”是DFS从树的角度和理解产生出来的概念,将不可能或者不确定的解剪掉,从而减少遍历的计算量,但要注意不要剪错了导致解缺失或错误

没有权值的树可以使用vector来表示,而不需要特意建立一个结构体,因为我们只是对树进行遍历而已

树的遍历需要掌握的大题://机试复试也可以看着复习;

  1. 遍历树,求出一条或多条权值和等于值W的路径

  1. //void DFS(int newNodeId, vector<int > path, int W)

  1. 求树的深度或者叶子结点的深度

//void DFS(int newNodeId, int depth);//递归来增加深度,son.depth == father + 1;

  1. 求结点的所在层次或者统计每层有多少个结点,最深的是那一层//BFS或者DFS都可以

// void DFS(int nowNodeId, int depth)

//

我们可以发现树的遍历DFS都是要当前的结点作为参数,且要用for去递归其孩子结点,这两步都是可以无脑写的,然后再看求什么再放什么参数,比如求深度就放个depth

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值