@Author:Runsen
@Date:2020/9/10
现在大四基本是重刷数据结构和算法,因为笔试真的太重要了。 Runsen又重温了争大佬专栏的队列,又巩固了下。而且Runsen发现留言区大佬的笔记很多,下面很多都是来自大佬总结的。
节点的高度=节点到叶子节点的最长路径(边数)
节点的深度=根节点到这个节点所经历的边的个数
节点的层数=节点的深度 + 1
树的高度=根节点的高度
1,二叉树,每个节点最多有两个叉,即两个子节点,分布是左子节点和右子节点。
2,二叉树不要求每个节点都有两个子节点,有的节点只有左子节点,有的节点只有右子节点。
3,满二叉树:叶子节点全部都在最底层,除了叶子节点之外,每个节点都有左右两个子节点,这种二叉树就叫做满二叉树。
4,完全二叉树:叶子节点都在最底下两层,最后一层的叶子节点都靠左排列,并且除了最后一层,其他层的节点个数都要达到最大。
5,二叉树的表示和储存方式:
①:存储一颗二叉树,有两种方法:
基于指针或者引用的二叉链表存储法;
基于数组的顺序存储法
②:链式存储法:
每个节点有三个字段,其中一个存储数据,另两个指向左右子节点的指针。这种存储方式比较常用。
③:顺序存储法
把根节点存储在下标i=1的位置,左子节点存储在下标 2i=2的位置,右子节点存储在2 * I +1=3的位置。以此类推。
如果节点X存储在数组中下标为i的位置,下标2i的位置存储的就是左子节点,下标为2*I +1的位置存储的就是右子节点。
反之,下标为i/2的位置存储就是他的父节点。通过这种方式,只要知道根节点存储位置,就可以通过下标计算,把整棵树都串起来。
使用数组储存二叉树,如果是一课完全二叉树,所以仅“浪费”一个下标为0的存储位置。如果是非完全二叉树,会浪费比较多的数组存储空间。
④:完全二叉树
如果某颗二叉树是一课完全二叉树,那用数组存储是最节省内存的一种方式。因为数组存储方式不需要像链式存储法那样,要存储额外的左右子节点的指针。这个是完全二叉树要求最后一层子节点都靠左的原因。
堆其实也是一种完全二叉树,最常用的存储方式就是数组。
二叉树遍历是非常常见的面试题,将所有节点都遍历打印出来的经典方法有三种:前序遍历,中序遍历,后序遍历。其中,前,中,后序,表示的是节点于它的左右子树节点遍历打印的先后顺序。
①:前序遍历:
对于树中的任意节点来将,先打印这个节点,然后在打印它的左子树,最后打印它的右子树。
②:中序遍历:
对于树中的任意节点来说,先打印它的左子树,然后在打印它本身,最后打印它的右子树。
③:后序遍历:
对于树中的任意节点来说,先打印它的左子树,然后在打印它的右子树,最后打印这个节点本身。
实际上,二叉树的前,中,后遍历就是一个递归的过程。
前序遍历的递推公式:
preOrder(r) = print r->preOrder(r->left)->preOrder(r->right)
具体实现代码:Leetcode 144. 二叉树的前序遍历
class Solution:
def preorderTraversal(self, root: TreeNode) -> List[int]:
res = []
def dfs(root):
if not root:
return []
res.append(root.val)
dfs(root.left)
dfs(root.right)
dfs(root)
return res
中序遍历的递推公式:
inOrder(r) = inOrder(r->left)->print r->inOrder(r->right)、
具体实现代码:Leetcode 94. 二叉树的中序遍历
class Solution:
def inorderTraversal(self, root: TreeNode) -> List[int]:
res = []
def dfs(root):
if not root:
return []
dfs(root.left)
res.append(root.val)
dfs(root.right)
dfs(root)
return res
后序遍历的递推公式:
postOrder(r) = postOrder(r->left)->postOrder(r->right)->print r
具体实现代码:Leetcode 145. 二叉树的后序遍历
class Solution:
def postorderTraversal(self, root: TreeNode) -> List[int]:
res = []
def dfs(root):
if not root:
return []
dfs(root.left)
dfs(root.right)
res.append(root.val)
dfs(root)
return res
每个节点最多会被访问两次,所以遍历操作的时间复杂度,更节点的个数n成正比,就是说二叉树遍历的时间复杂度是O(n)。
二叉树的遍历方式是最基本,也是最重要的一类题目,上面Runsen总结了「前序」、「中序」、「后序」、还剩一个「层序」遍历方式,方式就是使用双端队列。
下面具体二叉树层次遍历实现代码:Leetcode 102. 二叉树的层次遍历
class Solution:
def levelOrder(self, root: TreeNode) -> List[List[int]]:
'''
使用双端队列,方便两端元素入队出队操作
遍历每一层前,记录当前层节点数量 n
具体处理每一个节点时,若它有孩子,则将它的孩子入队
当前层遍历结束后立即将结果存储到最终结果中
'''
if not root:
return []
res,q = [],[root]
while q:
n =len(q)
level = []
for i in range(n):
node = q.pop(0)
level.append(node.val)
if node.left:
q.append(node.left)
# level.append(node.left)
if node.right:
q.append(node.right)
# level.append(node.right)
res.append(level)
return res
上面代码是bfs的广度优先搜索。下面补充下dfs深度优先搜索解决层次遍历的方法。
class Solution:
def __init__(self):
self.res = [] #存储最终结果的大列表
#增加一个参数表示层级,注意使用参数形式传递,这样子不会有全局干扰
def levelOrder(self, root: TreeNode,level=0) -> List[List[int]]:
#如果节点为空。直接返回一个空列表。根节点为空返回这个结果。子节点为空,相当于回退,不影响
if not root:
return []
# print((root.val,level)) #调试用
#如果当前层的子列表不存在,先新增一个空列表,后续再根据层值去插值
if len(self.res) <= level:
self.res.append([])
self.res[level].append(root.val)
level += 1 #更新层级
self.levelOrder(root.left,level)
self.levelOrder(root.right,level)
return self.res
在Leetcode中,Runsen发现了还有一个103. 二叉树的锯齿形层次遍历,也顺便搞定。
给定一个二叉树,返回其节点值的锯齿形层次遍历。(即先从左往右,再从右往左进行下一层遍历,以此类推,层与层之间交替进行)。
例如:
给定二叉树 [3,9,20,null,null,15,7],
3
/ \
9 20
/ \
15 7
返回锯齿形层次遍历如下:
[
[3],
[20,9],
[15,7]
]
这里和层次很像,就是多一个方向的判断,这里我直接用1表示右边,-1就是左边,这样使用列表切片可以反转。
# Definition for a binary tree node.
# class TreeNode:
# def __init__(self, x):
# self.val = x
# self.left = None
# self.right = None
class Solution:
def zigzagLevelOrder(self, root: TreeNode) -> List[List[int]]:
"""
:type root: TreeNode
:rtype: List[List[int]]
"""
if not root:
return []
res, level, direction = [], [root], 1
while level:
res.append([n.val for n in level][::direction])
direction *= -1
# 这里的level 需要去除上面res添加的
level = [kid for node in level for kid in (node.left, node.right) if kid]
return res
给定一组数据,比如 1,3,5,6,9,10。你来算算,可以构建出多少种不同的二叉树?
结果是卡特兰数,是C[n,2n] / (n+1)种形状,c是组合数,节点的不同又是一个全排列,一共就是 n ! ∗ C [ n , 2 n ] / ( n + 1 ) n!*C[n,2n] / (n+1) n!∗C[n,2n]/(n+1)个二叉树。可以通过数学归纳法推导得出。
确定两点:
1)n个数,即n个节点,能构造出多少种不同形态的树?
2)n个数,有多少种不同的排列?
当确定以上两点,将【1)的结果】乘以 【2)的结果】,即为最终的结果。
但是有一个注意的点: 如果n中有相等的数,产生的总排列数就不是 n ! n! n!了哟
通过这一题,Runsen学到了【卡塔兰数】:https://en.wikipedia.org/wiki/Catalan_number
补充:
卡塔兰数,通常记作c(n),是指n个A和n个B组成的这样的序列的数目,从前往后读,A的数目始终大于等于B。用组合数表示的话,c(n)=C(2n,n)/(n+1)。
卡特兰数的前10项分别为:1、2、5、14、42、132、429、1430、4862。
与斐波那契数列(Fibonacci Sequence)一样,卡塔兰数也是离散数学和编程算法中相当重要的数列,它以比利时的数学家欧仁·查理·卡塔兰(Eugene Charles Catalan,1814年~1894年)命名。
Runsen不是学这方面的,看到文章底部有人说起,就百度了解下。
参考:极客时间王争大佬:https://time.geekbang.org/column/article/67856