tree
每个结点最多有两个子树的树结构。通常子树被称作“左子树”(left subtree)和“右子树”(right subtree)。二叉树常被用于实现二叉查找树和二叉堆。
基本
父节点、子结点、兄弟结点
根节点:没有父节点的结点
叶子结点:没有子结点的结点
高度:节点到叶子结点的最长路径(边数)
深度:根节点到这个节点的边数
层数:节点深度 + 1
树的高度:根节点的高度。
二叉树(Binary Tree)是每个结点最多有两个子树的树结构
满二叉树
一棵深度为 k,且有 2^k-1 个节点的二叉树,称为满二叉树。
叶子节点全都在最底层,除了叶子节点之外,每个节点都有左右两个子节点。
完全二叉树
除最后一层外,若其余层都是满的,并且最后一层或者是满的,或者是在右边缺少连续若干节点,则此二叉树为完全二叉树。深度为 k 的完全二叉树,至少有 2^(k-1)个叶子节点,至多有 2^k-1 个节点。
具有 n 个节点的完全二叉树的深度为 floor(log2n)+1。
叶子节点都在最底下两层,最后一层的叶子节点都靠左排列,并且除了最后一层,其他层的节点个数都要达到最大。
对称二叉树
func isSymmetric(root *TreeNode) bool {
if root == nil {
return true
}
return dfs(root.Left, root.Right)
}
// 递归终止条件 两个节点都为空 两个节点中有一个为空 两个节点的值不相等
func dfs(left, right *TreeNode) bool {
if left == nil && right == nil {
return true
}
if left == nil || right == nil {
return false
}
if left.Val != right.Val {
return false
}
return dfs(left.Left, right.Right) && dfs(left.Right, right.Left)
}
翻转二叉树
递归
func invertTree(root *TreeNode) *TreeNode {
if root == nil {
return nil
}
left := invertTree(root.Left)
right := invertTree(root.Right)
root.Left = right
root.Right = left
return root
}
BFS
func invertTree(root *TreeNode) *TreeNode {
if root == nil {
return root
}
queue := []*TreeNode{root}
for len(queue) > 0 {
length := len(queue)
for i := 0; i < length; i++ {
node := queue[i]
node.Left, node.Right = node.Right, node.Left
if node.Left != nil {
queue = append(queue, node.Left)
}
if node.Right != nil {
queue = append(queue, node.Right)
}
}
queue = queue[length:]
}
return root
}
合并二叉树
func mergeTrees(t1 *TreeNode, t2 *TreeNode) *TreeNode {
if t1 == nil {
return t2
}
if t2 == nil {
return t1
}
t1.Val += t2.Val
t1.Left = mergeTrees(t1.Left, t2.Left)
t1.Right = mergeTrees(t1.Right, t2.Right)
return t1
}
遍历
- preorder 前序遍历 root-left-right
- inorder 中序遍历 left-root-right
- postorder 后序遍历 left-right-root
递归
func preorderTraversal(root *TreeNode) []int {
var res []int
var dfs func(*TreeNode)
dfs = func(root *TreeNode) {
if root == nil {
return
}
res = append(res, root.Val)
dfs(root.Left)
dfs(root.Right)
}
dfs(root)
return res
}
// inorderTraversal
dfs(root.Left)
res = append(res, root.Val)
dfs(root.Right)
// postorderTraversal
dfs(root.Left)
dfs(root.Right)
res = append(res, root.Val)
前序遍历
func preorderTraversal(root *TreeNode) (res []int) {
if root == nil {
return
}
stack := []*TreeNode{root}
for len(stack) > 0 {
top := stack[len(stack)-1]
stack = stack[:len(stack)-1]
res = append(res, top.Val)
if top.Right != nil {
stack = append(stack, top.Right)
}
if top.Left != nil {
stack = append(stack, top.Left)
}
}
return
}
中序遍历
func inorderTraversal(root *TreeNode) (res []int) {
var stack []*TreeNode
for root != nil || len(stack) > 0 {
for root != nil {
stack = append(stack, root)
root = root.Left
}
root = stack[len(stack)-1]
stack = stack[:len(stack)-1]
res = append(res, root.Val)
root = root.Right
}
return
}
后序遍历
- 中序遍历中,从栈中弹出的节点,其左子树是访问完了,可以直接访问该节点,然后接下来访问右子树。
- 后序遍历中,从栈中弹出的节点,我们只能确定其左子树肯定访问完了,但是无法确定右子树是否访问过。
因此,我们在后序遍历中,引入了一个 prev 来记录历史访问记录。
当访问完一棵子树的时候,我们用 prev 指向该节点。
这样,在回溯到父节点的时候,我们可以依据 prev 是指向左子节点,还是右子节点,来判断父节点的访问情况。
func postorderTraversal(root *TreeNode) (res []int) {
var prev *TreeNode
var stack []*TreeNode
for root != nil || len(stack) > 0 {
for root != nil {
stack = append(stack, root)
root = root.Left
}
root = stack[len(stack)-1]
stack = stack[:len(stack)-1]
// 没有右子树,或右子树已访问
if root.Right == nil || root.Right == prev {
res = append(res, root.Val)
prev = root
root = nil
} else {
// 右子树没有被访问,将当前节点压栈,访问右子树
stack = append(stack, root)
root = root.Right
}
}
return
}
层序遍历
func levelOrder(root *TreeNode) (res [][]int) {
if root == nil {
return
}
q := []*TreeNode{root}
level := 0
for len(q) > 0 {
length := len(q)
res = append(res, []int{})
for i := 0; i < length; i++ {
node := q[i]
res[level] = append(res[level], node.Val)
if node.Left != nil {
q = append(q, node.Left)
}
if node.Right != nil {
q = append(q, node.Right)
}
}
q = q[length:]
level++
}
return
}
从前序与中序遍历序列构造二叉树
题目说,假设树中没有重复的元素
func buildTree(preorder []int, inorder []int) *TreeNode {
// 用于快速定位根节点
indexMap := make(map[int]int, len(inorder))
for i, v := range inorder {
indexMap[v] = i
}
var dfs func(int, int, int) *TreeNode
dfs = func(preStart, preEnd, inStart int) *TreeNode {
if preStart > preEnd {
return nil
}
// 前序遍历的第一个节点就是根节点
root := &TreeNode{Val: preorder[preStart]}
// 在中序遍历中定位根节点
rootIdx := indexMap[preorder[preStart]]
// 左子树中的节点数目
leftSize := rootIdx - inStart
root.Left = dfs(preStart+1, preStart+leftSize, inStart)
root.Right = dfs(preStart+leftSize+1, preEnd, rootIdx+1)
return root
}
return dfs(0, len(preorder)-1, 0)
}
之字形层序遍历
自底向上的层序遍历
func levelOrderBottom(root *TreeNode) [][]int {
res := make([][]int, 0)
if root == nil {
return res
}
queue := []*TreeNode{root}
level := 0
for len(queue) > 0 {
length := len(queue)
res = append(res, []int{})
for i := 0; i < length; i++ {
node := queue[i]
res[level] = append(res[level], node.Val)
if node.Left != nil {
queue = append(queue, node.Left)
}
if node.Right != nil {
queue = append(queue, node.Right)
}
}
queue = queue[length:]
level++
}
// 反转数组
L, R := 0, len(res)-1
for L < R {
res[L], res[R] = res[R], res[L]
L++
R--
}
return res
}
二叉树的最大深度
func maxDepth(root *TreeNode) int {
if root == nil {
return 0
}
return max(maxDepth(root.Left), maxDepth(root.Right)) + 1
}
func max(x, y int) int {
if x > y {
return x
}
return y
}
二叉树的最小深度
DFS
func minDepth(root *TreeNode) int {
if root == nil {
return 0
}
if root.Left == nil && root.Right == nil {
return 1
}
min := math.MaxInt32
if root.Left != nil {
min = minInt(minDepth(root.Left), min)
}
if root.Right != nil {
min = minInt(minDepth(root.Right), min)
}
return min + 1
}
func minInt(x, y int) int {
if y < x {
return y
}
return x
}
BFS
当我们找到一个叶子节点时,直接返回这个叶子节点的深度。广度优先搜索的性质保证了最先搜索到的叶子节点的深度一定最小。
func minDepth(root *TreeNode) int {
if root == nil {
return 0
}
queue := []*TreeNode{root}
level := 1
for len(queue) > 0 {
length := len(queue)
for i := 0; i < length; i++ {
if queue[i].Left == nil && queue[i].Right == nil {
return level
}
if queue[i].Left != nil {
queue = append(queue, queue[i].Left)
}
if queue[i].Right != nil {
queue = append(queue, queue[i].Right)
}
}
queue = queue[length:]
level++
}
return level
}
二叉树的堂兄弟节点
func isCousins(root *TreeNode, x, y int) bool {
var xParent, yParent *TreeNode
var xDepth, yDepth int
var xFound, yFound bool
var dfs func(node, parent *TreeNode, depth int)
dfs = func(node, parent *TreeNode, depth int) {
if node == nil {
return
}
if node.Val == x {
xParent, xDepth, xFound = parent, depth, true
} else if node.Val == y {
yParent, yDepth, yFound = parent, depth, true
}
// 如果两个节点都找到了,就可以提前退出遍历
// 即使不提前退出,对最坏情况下的时间复杂度也不会有影响
if xFound && yFound {
return
}
dfs(node.Left, node, depth+1)
if xFound && yFound {
return
}
dfs(node.Right, node, depth+1)
}
dfs(root, nil, 0)
return xDepth == yDepth && xParent != yParent
}
N 叉树的最大深度
func maxDepth(root *Node) int {
if root == nil {
return 0
}
max := 0
for _, child := range root.Children {
max = int(math.Max(float64(max), float64(maxDepth(child))))
}
return max + 1
}
N 叉树的前序遍历
func preorder(root *Node) (res []int) {
if root == nil {
return
}
s := []*Node{root}
for len(s) > 0 {
node := s[len(s)-1]
s = s[:len(s)-1]
res = append(res, node.Val)
for i := len(node.Children) - 1; i >= 0; i-- {
s = append(s, node.Children[i])
}
}
return
}
N 叉树的后序遍历
func postorder(root *Node) (res []int) {
if root == nil {
return
}
s := []*Node{root}
for len(s) > 0 {
node := s[len(s)-1]
s = s[:len(s)-1]
res = append(res, node.Val)
for _, child := range node.Children {
s = append(s, child)
}
}
length := len(res)
for i := 0; i < length/2; i++ {
res[i], res[length-1-i] = res[length-1-i], res[i]
}
return
}
N 叉树的层序遍历
func levelOrder(root *Node) (ans [][]int) {
if root == nil {
return
}
q := []*Node{root}
for len(q) > 0 {
var level []int
length := len(q)
for i := 0; i < length; i++ {
level = append(level, q[i].Val)
q = append(q, q[i].Children...)
}
ans = append(ans, level)
q = q[length:]
}
return
}
Morris Traversal
不使用栈,使用线索二叉树,空间复杂度为 O(1)
每个前驱恰好访问两次,前序遍历和中序遍历仅有一行代码不同
func inorderTraversal(root *TreeNode) []int {
var res []int
curr := root
for curr != nil {
if curr.Left == nil {
res = append(res, curr.Val)
curr = curr.Right
} else {
pre := curr.Left
for pre.Right != nil && pre.Right != curr {
pre = pre.Right
}
if pre.Right == nil {
pre.Right = curr
curr = curr.Left
} else {
res = append(res, curr.Val)
pre.Right = nil
curr = curr.Right
}
}
}
return res
}
func preorderTraversal(root *TreeNode) []int {
var res []int
curr := root
for curr != nil {
if curr.Left == nil {
res = append(res, curr.Val)
curr = curr.Right
} else {
pre := curr.Left
for pre.Right != nil && pre.Right != curr {
pre = pre.Right
}
if pre.Right == nil {
res = append(res, curr.Val)
pre.Right = curr
curr = curr.Left
} else {
pre.Right = nil
curr = curr.Right
}
}
}
return res
}
BST
高度平衡 二叉树是一棵满足「每个节点的左右两个子树的高度差的绝对值不超过 1 」的二叉树。
二叉查找树(Binary Search Tree),它或者是一棵空树,或者是具有下列性质的二叉树:
- 若它的左子树不空,则左子树上所有结点的值均小于它的根结点的值;
- 若它的右子树不空,则右子树上所有结点的值均大于它的根结点的值;
- 它的左、右子树也分别为二叉排序树。
二叉搜索树作为一种经典的数据结构,它既有链表的快速插入与删除操作的特点,又有数组快速查找的优势; 所以应用十分广泛,例如在文件系统和数据库系统一般会采用这种数据结构进行高效率的排序与检索操作。
验证二叉搜索树
func isValidBST(root *TreeNode) bool {
max := math.MinInt64
var stack []*TreeNode
for root != nil || len(stack) > 0 {
for root != nil {
stack = append(stack, root)
root = root.Left
}
root = stack[len(stack)-1]
stack = stack[:len(stack)-1]
if root.Val <= max {
return false
}
max = root.Val
root = root.Right
}
return true
}
有序数组转二叉搜索树
func sortedArrayToBST(nums []int) *TreeNode {
if len(nums) == 0 {
return nil
}
mid := len(nums) / 2
return &TreeNode{nums[mid], sortedArrayToBST(nums[:mid]), sortedArrayToBST(nums[mid+1:])}
}
二叉搜索树中的搜索
递归
func searchBST(root *TreeNode, val int) *TreeNode {
if root == nil || root.Val == val {
return root
}
if root.Val > val {
return searchBST(root.Left, val)
}
return searchBST(root.Right, val)
}
迭代
func searchBST(root *TreeNode, val int) *TreeNode {
for root != nil && root.Val != val {
if root.Val > val {
root = root.Left
} else {
root = root.Right
}
}
return root
}
Trie
实现 Trie (前缀树)
type Trie struct {
children map[rune]*Trie
isEnd bool
}
func Constructor() Trie {
return Trie{make(map[rune]*Trie), false}
}
func (trie *Trie) Insert(word string) {
p := trie
for _, ch := range word {
if child, ok := p.children[ch]; ok {
p = child
} else {
newNode := &Trie{children: make(map[rune]*Trie)}
p.children[ch] = newNode
p = newNode
}
}
p.isEnd = true
}
func (trie *Trie) Search(word string) bool {
p := trie
for _, ch := range word {
if child, ok := p.children[ch]; ok {
p = child
continue
}
return false
}
return p.isEnd
}
func (trie *Trie) StartsWith(prefix string) bool {
p := trie
for _, ch := range prefix {
if child, ok := p.children[ch]; ok {
p = child
continue
}
return false
}
return true
}