数据结构与算法(九) 树

简介: 数据结构与算法(九) 树

前言


大家好,我是半虹,这篇文章来讲树

树是一种常见的数据结构,其定义为:由有限个节点组成的具有层次关系的集合

解决树问题的关键是递归,递归的关键是分解子问题

对于树来说,递归函数只要考虑对单个节点如何处理,然后通过递归调用拓展到整棵树即可

下面具体介绍一些常见的树结构及相关算法,以深入地理解如何应用递归解决树相关的问题


正文


1  二叉树


二叉树是特殊的树结构,每个节点最多有两个子节点,分别称为左孩子和右孩子

二叉树很多的相关问题,基本上都可以用递归来解决,下面具体来看一下怎么用


1.1 初识递归

对于一个数据结构来说,最基本的操作就是遍历元素,二叉树的遍历同样是基于递归的

而其中最主要的问题就是怎么设计递归函数,为此可将二叉树抽象成下面的模式

先.png

为什么是这样的模式呢,这其实就是二叉树的子结构,在递归中可理解成子问题

递归函数要做的事情就是解决好这个子问题,然后再通过递归调用当左右孩子是新的父节点拓展到整棵树


这个是解决二叉树问题的一种常用思考模式,是从上往下的理解,而写递归逻辑时也可以从下往上去理解


设递归函数能分别处理以左右孩子为根节点的树对应的题目要求,也即可以通过递归处理左子树和右子树


则能否处理以父节点为根节点的树所对应的题目要求,如果是能,具体如何操作,写出来就是递归的逻辑


这样表达还是有些抽象,下面就以遍历为例具体讲讲


现题目要求遍历二叉树,我们假设能通过递归调用分别遍历以左右孩子为根节点的树

那么要遍历以父节点为根节点的树很简单,只需访问父节点,然后通过递归调用遍历左子树和右子树即可

将这些操作都写出来就是递归函数的逻辑,对应的代码如下:

// 递归函数:只需考虑如何去处理子结构
// 功能定义:遍历以父节点为根节点的树
void traverse(TreeNode* root) {
    // 边界情况
    // 这里对应的是当父节点是空指针的情况
    // 此时无需进行任何操作,直接返回就行
    if (root == null) {
        return;
    }
    // 基本情况
    // 这里对应的是当父节点是树节点的情况
    cout  << root -> val << endl; // 访问父节点
    traverse(root -> left);       // 遍历左子树(递归调用)
    traverse(root -> right);      // 遍历右子树(递归调用)
}


看递归函数千万不要跟着深入,只需要定义好递归函数的功能,知道调用函数能完成什么事就可以

上述的函数所做的事情很简单:打印传入节点,然后将该节点的左孩子和右孩子作为参数递归调用

递归就是这样的,把完整过程想清楚很难,把子问题处理好很简单,关键是怎么定义和处理子问题


在上述代码中,对访问父节点的位置也有讲究,不同位置对应着二叉树的三种遍历方式,具体如下:


前序遍历:先访问父节点,再访问左孩子,后访问右孩子(前序指父节点首先访问)

中序遍历:先访问左孩子,再访问父节点,后访问右孩子(中序指父节点中间访问)

后序遍历:先访问左孩子,再访问右孩子,后访问父节点(后序指父节点最后访问)


对应例子图示如下:

法.png

对应代码框架如下:

void traverse(TreeNode* root) {
    if (root == null) {
        return;
    }
    // 在这里访问父节点,就是前序遍历
    traverse(root -> left);     // 遍历左子树(递归调用)
    // 在这里访问父节点,就是中序遍历
    traverse(root -> right);    // 遍历右子树(递归调用)
    // 在这里访问父节点,就是后序遍历
}


大家不要小看遍历,这是解决二叉树问题的一种基础模式,一个典型的例题就是序列化与反序列化

所谓序列化就是将二叉树变成序列,其实就是将二叉树的遍历结果存下来就可以

而反序列化则是将序列变回二叉树,这个就是根据序列用递归的方式还原二叉树

用三种遍历方式来做序列化都可以,下面我们就用前序遍历举例给出代码,leetcode297

class Codec {
public:
    // 序列化
    // 递归函数:序列化以传入节点为根节点的子树
    string serialize(TreeNode* root) {
        // 边界情况
        // 对应的是父节点是空指针的情况
        // 此时要将父节点转字符串,但是不用处理左右子树
        if (!root) {
            return "#";
        }
        // 基本情况
        // 对应的是父节点是树节点的情况
        // 此时要将父节点转字符串,同时通过递归调用来将左右子树序列化
        return to_string(root -> val) + "," + 
               serialize(root -> left) + "," +
               serialize(root -> right);
    }
    // 反序列化
    TreeNode* deserialize(string data) {
        // 先转列表
        string temp;
        list<string> dataList;
        for (char& ch: data) {
            if (ch == ',') {
                dataList.push_back(temp);
                temp.clear();
            } else {
                temp += ch;
            }
        }
        if (!temp.empty()) {
            dataList.push_back(temp);
            temp.clear();
        }
        // 进入递归
        return my_deserialize(dataList);
    }
    // 递归函数:构造以传入列表第一个元素为根节点的子树
    // 这里注意,参数列表是动态变化的
    TreeNode*  my_deserialize(list<string>& dataList) {
        // 边界情况
        // 若当元素为空指针
        // 则直接返回空指针,且不用处理左右子树
        if (dataList.front() == "#") {
            dataList.erase(dataList.begin());
            return nullptr;
        }
        // 基本情况
        // 若当元素为树节点
        // 则将其作为父节点,并通过递归调用设置左右子树
        TreeNode* root = new TreeNode(stoi(dataList.front()));
        dataList.erase(dataList.begin());
        root -> left = my_deserialize(dataList);
        root -> right = my_deserialize(dataList);
        return root;
    }
};


序列化时需要把空指针也记录下来,这样后面才能去还原,没有空指针标记是无法分辨树结构的

没有空指针标记,但已知中序遍历和前序遍历,或已知中序遍历和后序遍历,也可以还原二叉树

但是这里要注意,若已知前序遍历和后序遍历,并不能唯一确定一棵二叉树

下面假设已知中序遍历和前序遍历,给出还原二叉树的代码如下,leetcode105

// 事实
// 前序遍历结果:[父节点, [左子树前序遍历结果], [右子树前序遍历结果]]
// 中序遍历结果:[[左子树中序遍历结果], 父节点, [右子树中序遍历结果]]
// 
// 思路
// 1. 从前序遍历结果可知父节点,很显然第一个值就是
// 2. 从中序遍历结果搜索父节点,父节点左边的值就是左子树,父节点右边的值就是右子树,然后再统计左右子树元素数量
// 3. 然后根据左右子树元素数量,从前序遍历结果分出左子树和右子树
// 4. 最后再对左右子树执行上述的步骤(递归)
class Solution {
public:
    unordered_map<int, int> val2idx;
    TreeNode* buildTree(vector<int>& preorder, vector<int>& inorder) {
        // 构造哈希映射
        // 可加速在中序遍历列表找到根节点
        for (int i = 0; i < inorder.size(); i++) {
            val2idx[inorder[i]] = i;
        }
        // 进入递归
        return myBuildTree(
            preorder,
            inorder,
            0,
            preorder.size() - 1,
            0,
            inorder.size() - 1
        );
    }
    // 递归函数
    // 构造子树
    TreeNode* myBuildTree(
        vector<int>& preorder,
        vector<int>& inorder,
        // 不修改原序列
        // 通过下标表示要处理的子序列
        int preorder_left,
        int preorder_right,
        int inorder_left,
        int inorder_right
    ) {
        // 边界情况
        if (preorder_left > preorder_right) {
            return nullptr;
        }
        // 基本情况
        int root_value = preorder[preorder_left];
        TreeNode* root = new TreeNode(root_value); // 父节点
        int inorder_root = val2idx[root_value];
        int left_subtree = inorder_root - inorder_left;
        root -> left = myBuildTree( // 为父节点构造左子树
            preorder,
            inorder,
            preorder_left + 1,
            preorder_left + left_subtree,
            inorder_left,
            inorder_root - 1
        );
        root -> right = myBuildTree( // 为父节点构造右子树
            preorder,
            inorder,
            preorder_left + 1 + left_subtree,
            preorder_right,
            inorder_root + 1,
            inorder_right
        );
        return root;
    }
};


1.2 深入递归


上面我们主要讲了三种遍历方式和怎么根据遍历结果还原二叉树,解决这些问题都要用到递归的思想


下面我们再来看些更复杂的递归,并探讨如何在递归中传递数据,概括地说会有两种传递数据的方式


一是通过递归函数的返回值传递,二是通过递归函数的参数或者是外部变量辅助传递


但无论是哪种方式,解递归问题的关键依然是定义和处理子问题,下面通过例题介绍


(1)二叉树的最大深度 | leetcode104


给定二叉树,计算其最大深度,二叉树的深度定义为根节点到最远叶子节点的路径上的节点数


用从下往上的角度来思考问题,设递归函数可以分别求出以左右孩子为根节点的树的最大深度


则求出以父节点为根节点的树的最大深度也很简单,只要用左右子树最大深度中的较大值加上自己就可以


在这里,最大深度的信息是从下往上传的,具体来说是通过递归函数的返回值来传的


所以要先计算子树的最大深度后才能计算原树的最大深度,也就是说对原树的计算要放在后序位置去执行

class Solution {
public:
    int maxDepth(TreeNode* root) {
        // 边界情况
        if (!root) {
            return 0;
        }
        // 基本情况
        int left = maxDepth(root -> left); // 左子树最大深度
        int right = maxDepth(root -> right); // 右子树最大深度
        int ans = max(left, right) + 1; // 后序位置
        return ans;
    }
};


用从上往下的角度来思考问题,先定义一个递归函数,接收一个树节点作为参数


其功能定义为计算传入树节点的深度,然后递归调用该函数计算左孩子和右孩子的深度


而左右孩子的深度其实就等于父节点的深度加一


这样当我们初始传入根节点时,该函数就能得到所有树节点的深度,然后取最大值就是树的最大深度


这里,最大深度的信息是从上往下传的,具体来说是通过递归函数的参数来传的


所以要先有父节点的深度才能计算左右孩子的深度,也就是说对父节点的计算要在前序位置执行

另外,在从上往下传当前深度信息时,同时用外部变量保存最大深度信息

class Solution {
public:
    int ansDepth = 0;
    int maxDepth(TreeNode* root) {
        traverse(root, 1);
        return ansDepth;
    }
    void traverse(TreeNode* root, int curDepth) {
        // 边界情况
        if (!root) {
            return;
        }
        // 基础情况
        // 由于在传入参数时就同时传入了节点的深度
        // 其实也就是在前序位置时就处理好了父节点
        ansDepth = max(ansDepth , curDepth);
        traverse(root -> left, curDepth + 1); // 遍历左子树
        traverse(root -> right, curDepth + 1); // 遍历右子树
    }
};


(2)二叉树的最大路径和 | leetcode124

给定二叉树,计算其最大路径和,路径和也就是路径中各个节点值的总和

路径定义为从树中任意节点出发达到任意节点的序列,要求同一个节点在一条路径中至多出现一次

例如给定二叉树如下:
     -10
     /  \
    9   20
       /  \
      15   7
那么最大路径和就是:42,对应的路径就是:15 -> 20 -> 7


上面对路径的定义很难用于解题,我们首先试一下能不能转化为二叉树的基本模式,也就是父左右

每一条路径可看作是有一个顶点,也即父节点,其既可与左路径或者与右路径连接,也可单独存在

第一种情况:只有顶点
               n1 (顶点)
第二种情况:顶点只与左路径连接
               n1 (顶点)
              /
             n2
             |
            ...
第三种情况:顶点只与右路径连接
               n1 (顶点)
                 \
                 n3
                  |
                 ...
第四种情况:顶点既与左路径连接、又与右路径连接
               n1 (顶点)
              /  \
             n2  n3
             |    |
            ...  ...


按照先前的经验,用从下往上的角度思考问题,设递归函数能分别求出以左右孩子为顶点的最大路径和

那么能否求出以父节点为顶点的最大路径和呢?好像不太行,主要问题出现在第四种情况

若左右孩子所已知的最大路径和所对应的路径是第四种情况
那么父节点无论是连接左孩子还是右孩子都是不合法的路径
例如左孩子所已知的最大路径和所对应的路径如下所示:n4 -> n2 -> n5
               n2
              /  \
             n4  n5
那么父节点连接左孩子的路径是不合法的,因为没办法用不重复的序列穿起四个节点
                 n1
                /
               n2
              /  \
             n4  n5


那要怎么处理呢?别急,我们先来看一个简化版题目,那就是二叉树的单边最大路径和


在这里,路径定义简化为从父节点到子节点的单路径,也就是先忽略上面的第四种情况


处理这种情况就很简单,先设递归函数能分别求出以左右孩子为顶点的单边最大路径和


则求出以父节点为顶点的单边最大路径和只要考虑以下情况的最大值:父、父加左路径、父加右路径


class Solution {
public:
    // 单边最大路径和
    int maxSideSum(TreeNode* root) {
        // 边界情况
        if (!root) {
            return 0;
        }
        // 基本情况
        int left = maxSideSum(root -> left); // 以左孩子为顶点的单边最大路径和
        int right = maxSideSum(root -> right); // 以右孩子为顶点的单边最大路径和
        int val = root -> val;
        int cur = max({val, val + left, val + right});
        return cur;
    }
};

现在我们顺着这个思路再深入一下

假设我们知道以左右孩子为顶点的单边最大路径和,能不能求出以父节点为顶点的最大路径和呢

好像也是可以的,只要考虑以下情况的最大值:父、父加左路径、父加右路径、父加左路径加右路径

为此我们可以基于上面的递归框架,多加一步来计算最大路径和就可以

为此我们可以基于上面的递归框架,多加一步来计算最大路径和就可以
class Solution 


这里需要注意,计算最大路径和这一步是附加在单边最大路径和的递归逻辑上的,并不影响原有的递归逻辑

只是刚好在解决单边最大路径和时,可以顺便计算出最大路径和,这给我们一个解题的启发

那就是在原有的设定下无法递归时,可以转为递归简化版的问题,然后从中计算原问题的解


(3)二叉树的最近公共祖先 | leetcode236

给定二叉树和两个树节点,找出这两个节点的最近公共祖先

最近公共祖先要求为同时是两个节点的祖先且深度尽可能大

例如给定二叉树如下:
         3
       /   \
     5       1
    / \     / \
   6   2   0   8
      / \
     7   4
节点 5 和节点 1 的最近公共祖先是节点 3
节点 5 和节点 4 的最近公共祖先是节点 5

解决这个问题有一个很直观的方法,那就是遍历一次二叉树,记录每个节点的深度和父节点

然后对于给定的两个树节点,每次将深度较大的节点指向其父节点,直至二者汇合就是最近公共祖先

class Solution {
public:
    unordered_map<TreeNode*, int> depth;
    unordered_map<TreeNode*, TreeNode*> parent;
    TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {
        depth[root] = 1;
        parent[root] = nullptr;
        traverse(root, 1);
        while  (true) {
            if (p == q) {
                return p;
            }
            if (depth[p] == depth[q]) {
                p = parent[p];
                q = parent[q];
            }
            else if (depth[p] > depth[q]) {
                p = parent[p];
            }
            else if (depth[p] < depth[q]) {
                q = parent[q];
            }
        }
    }
    void traverse(TreeNode* root, int curDepth) {
        if (root -> left != nullptr) {
            depth[root -> left] = curDepth + 1;
            parent[root -> left] = root;
            traverse(root -> left, curDepth + 1);
        }
        if (root -> right != nullptr) {
            depth[root -> right] = curDepth + 1;
            parent[root -> right] = root;
            traverse(root -> right, curDepth + 1);
        }
    }
};


然而这个问题就不能用递归的方法来做吗?当然可以,虽然咋一看好像很难去定义递归函数

我们不妨思考下究竟什么是最近公共祖先,一个节点是最近公共祖先必须满足以下条件之一

  1. 该节点的左右子树分别包含给定的两个节点
  2. 该节点的左右子树包含给定的两个节点之一,且该节点本身就是另一个给定的节点

为此我们可以定义递归函数功能为判断以父节点为根节点的树是否包含两个给定的节点中的至少一个

验证一下这个定义是否可求,设递归函数能判断以左右孩子为根节点的树是否包含给定节点

那么判断以父节点为根节点的树是否包含给定节点非常简单

当父节点本身就是给定节点、或其左、右子树包含给定节点时为真,其余情况为假

class Solution {
public:
    // 判断以 root 为根节点的树是否包含 p 或 q 节点
    bool haveNode(TreeNode* root, TreeNode* p, TreeNode* q) {
        // 边界情况
        if (!root) {
            return false;
        }
        // 基础情况
        bool left = haveNode(root -> left, p, q); // 左子树是否包含给定节点
        bool right = haveNode(root -> right, p, q); // 右子树是否包含给定节点
        return left || right || root -> val == p -> val || root -> val == q -> val;
    }
};


那么能否在此基础上求出最近公共祖先呢?假设已知左右子树是否包含给定节点

在判断父节点是否为最近公共祖先时,只需按照最近公共祖先的两条件判断即可

class Solution {
public:
    TreeNode* ans;
    TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {
        haveNode(root, p, q);
        return ans;
    }
    // 判断以 root 为根节点的树是否包含 p 或 q 节点
    bool haveNode(TreeNode* root, TreeNode* p, TreeNode* q) {
        if (!root) {
            return false;
        }
        bool left = haveNode(root -> left, p, q);
        bool right = haveNode(root -> right, p, q);
        if ( // 新增
            (left && right) || // 满足条件 1
            ((root -> val == p -> val || root -> val == q -> val) && (left || right)) // 满足条件 2
        ) {
            ans = root;
        }
        return left || right || root -> val == p -> val || root -> val == q -> val;
    }
};


这就跟上一道题目是一样的,都是因为按原设定无法递归,所以选择递归其变种,并从中求出原问题的答案


又或者我们从另一个角度来看一下上述的解法,递归函数在后序位置返回值实际上就是从下往上传递的数据


这整个传递过程从叶子节点开始到根节点结束,而根节点的返回值就是整个递归函数的最终返回值


以上述递归函数来举例,该函数返回的是布尔,其表示以当前节点为根节点的树是否包含给定节点


而在此基础上,该函数在传递值到每个节点时,会根据当前的信息判断该节点是否为最近公共祖先


判断条件设置很巧妙,所有节点有且仅有一个符合要求,然后通过外部变量保存该节点,就是答案

例如给定二叉树如下,其中的节点 6 和 7 是给定节点,现在要求找出这两个节点的最近公共祖先
             3(T)             /|\
           /      \            |
      {5}(T)       1(F)        |
      /    \      /   \        |
   [6](T)  2(T) 0(F)  8(F)     |
          /   \                |
      [7](T)  4(F)             |
这其中 :
小括号 () 表示递归函数传递的数据,从下往上传递,汇合到根节点
中括号 [] 表示两个给定节点
大括号 {} 表示最近公共祖先

按照这个思路,其实我们可以直接设计出一个递归函数,将树节点作为返回值,使其最终返回最近公共祖先


返回规则如下:


当父节点为空的指针时,返回空的指针(边界情况)

当父节点为给定节点时,返回给定节点

当父节点为其余节点时,如果其左子树和右子树都包含给定节点,返回父节点,就是最近公共祖先

当父节点为其余节点时,如果其左子树或右子树有包含给定节点,返回该给定节点

当父节点为其余节点时,如果其左子树和右子树没包含给定节点,返回空指针


直观图示如下:

例如给定二叉树如下,其中的节点 6 和 7 是给定节点,现在要求找出这两个节点的最近公共祖先
             3(5)             /|\
           /      \            |
      {5}(5)       1(n)        |
      /    \      /   \        |
   [6](6)  2(7) 0(n)  8(n)     |
          /   \                |
      [7](7)  4(n)             |
这其中 :
小括号 () 表示函数的返回值,也就是传递的数据,数字 表示树节点指针,n 表示空指针
中括号 [] 表示两个给定节点
大括号 {} 表示最近公共祖先
符合返回规则 1 的节点有:/
符合返回规则 2 的节点有:6、7
符合返回规则 3 的节点有:5
符合返回规则 4 的节点有:2、3
符合返回规则 5 的节点有:4、0、8、1

对应代码如下:

class Solution {
public:
    TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {
        // 返回规则 1
        if (root == nullptr) {
            return  nullptr;
        }
        // 返回规则 2
        if (root == p) {
            return  p;
        }
        if (root == q) {
            return  q;
        }
        // 左右子树 返回值
        TreeNode* left = lowestCommonAncestor(root -> left, p, q);
        TreeNode* right = lowestCommonAncestor(root -> right, p, q);
        // 返回规则 3
        if (left && right) {
            return root;
        }
        // 返回规则 4
        else if (left || !right) {
            return left;
        }
        else if (!left || right) {
            return right;
        }
        // 返回规则 5
        else {
            return nullptr;
        }
    }
};


1.3 小结


与二叉树相关的问题,基本上都可以用递归解决,而递归的关键就是分解子问题

对二叉树而言,存在着一种很典型的子问题模式,也就是上面提到的父左右模式

以后遇到二叉树问题,首先可以往这种模式靠拢


假如确定二叉树问题能符合上述的模式,则可以参考下面提供的思路来解决问题

首先要将父左右模式放在最普遍的场景下去思考

出.png


然后再定义递归函数,思考怎么去处理以当前父节点为根节点的树,使其能满足题目要求


若当前父节点要获取其父节点的数据,则通过递归函数的参数传递下来


且当前父节点在调用递归函数处理其子节点时需要传入与接收参数定义一致的数据


若当前父节点要获取其子节点的数据,则通过递归函数返回值传递上来


且当前父节点的处理逻辑需要在函数的最后去返回与所接收的内容定义一致的数据


至此,已定义好递归函数的所有内容,最后再提醒大家注意下三个细节


在已定义好递归函数后,调用递归函数时需要相信就是能做到你所定义的事情,别跟递归调用层层深入

在写父节点处理逻辑时,分类讨论节点的不同情况,例如当节点是空指针或是树指针时的处理是不同的

在对父节点传递数据时,定义递归函数的参数和递归函数返回值所代表的含义,这样会使逻辑更加清晰


下面是用伪代码所写的框架,后面会有好几道例题,帮助大家进行回顾

BottomUpData function(root, TopDownData) {
    if (root == nullptr) {
        // ...
    }
    else if (root == TreeNode) {
        // ...
        TopDownDataToLeft = process1(root, TopDownData);
        TopDownDataToRight = process1(root, TopDownData);
        // ...
        BottomUpDataFromLeft = function(root -> left, TopDownDataToLeft);
        BottomUpDataFromRight = function(root -> right, TopDownDataToRight);
        // ...
        return process2(root, BottomUpDataFromLeft, BottomUpDataFromRight);
    }
    // else if (...) {
    //     ...
    // }
}


注意,上述所说的所有内容,都是针对父左右的递归模式来说的


但是,并非所有二叉树问题,都会符合这种模式,而此时可试着分析二叉树中的其他子问题


另外,更非所有二叉树问题,都能使用递归去做,若发现不能用递归解决,则要另外找方法


总之,解决问题应是灵活的,千万不要被框架困住了思维


好了,该说的都已经说完了,下面来几道题目检查下学习成果吧



(1)二叉树展开为链表 | leetcode114

给定二叉树将其展开为链表,要求展开后的链表与二叉树的前序遍历一致

链表的元素是树节点,节点的左孩子指向空指针,右孩子指向下一个元素


解题思路如下:

假设给定二叉树如下:
         6
       /   \
     2       8
    / \     / \
   1   4   7   9
      / \
     3   5
前序遍历顺序为:[6, [左子树前序遍历结果], [右子树前序遍历结果]]
显然,
左子树前序遍历的倒数第一个元素是:5(左子树最下最右的节点)
右子树前序遍历的正数第一个元素是:8(右子树的根节点)
所以,
对于每个根节点我们可以做如下处理:
1. 将右子树前序遍历正数第一个的元素接到左子树前序遍历倒数第一个的元素的右孩子
         6
       /
     2
    / \
   1   4
      / \
     3   5
          \
           8
          / \
         7   9
2. 将左子树接到父节点的右孩子
         6
           \
            2
           / \
          1   4
             / \
            3   5
                 \
                  8
                 / \
                7   9
3. 至此,已完成父节点处理逻辑,之后再递归调用处理右孩子即可
   此时,树的前序遍历结果不变


对应代码如下:

class Solution {
public:
    void flatten(TreeNode* root) {
        if (root == nullptr) {
            // 如果根节点为空,则直接返回
            return;
        }
        else if (root -> left == nullptr && root -> right == nullptr) {
            // 如果左孩子为空且右孩子为空,则直接返回
            return;
        }
        else if (root -> left == nullptr && root -> right != nullptr) {
            // 如果左孩子为空但右孩子不空,则递归调用处理右孩子
            flatten(root -> right);
        }
        else if (root -> left != nullptr && root -> right == nullptr) {
            // 如果左孩子不空但右孩子为空,则将左孩子移到右孩子,再递归调用处理右孩子
            root -> right = root -> left;
            root -> left = nullptr;
            flatten(root -> right);
        }
        else if (root -> left != nullptr && root -> right != nullptr) {
            // 如果左孩子不空且右孩子不空,则将左子树接到右子树,再将左孩子移到右孩子,再递归调用处理右孩子
            TreeNode* last_in_left = root -> left;
            while (last_in_left -> right != nullptr) {
                last_in_left = last_in_left -> right;
            }
            last_in_left -> right = root -> right;
            root -> right = root -> left;
            root -> left = nullptr;
            flatten(root -> right);
        }
    }
};


(2)不同的二叉搜索树 | leetcode96


给定整数 n,求恰由 n 个节点组成且节点值从 1 到 n 的不同二叉搜索树数量



二叉搜索树有一个性质,每个节点的值大于等于其左子树所有节点的值且小于等于其右子树所有节点的值


如果将数组的某个元素作为父节点,那么只能用该元素左边值构造左子树,并用该元素右边值构造右子树


而对于左右子树,也可以用同样的方法去处理,这就是子问题,看个例子:

假设给定整数 9 ,那么对应二叉树节点为 [1, 2, 3, 4, 5, 6, 7, 8, 9]
然后我们可以选择任意一个节点作为根节点 构造二叉搜索树
例如选择元素 7 作为根节点,那么只能用 [1, 2, 3, 4, 5, 6] 作为左子树,用 [8, 9] 作为右子树
接下来只需要递归调用处理左右子树就可以
边界条件是:当子树只有零个或一个节点时 其构造方法只有一种
---
另外注意到,在上述求解时存在重叠子问题
接上面举例,继续处理左子树 [1, 2, 3, 4, 5, 6]
如果选择元素 3 作为根节点,那么,其左子树就是 [1, 2],其右子树就是 [4, 5, 6]
而用 [1, 2] 构造子树,和用 [8, 9] 构造子树,结果其实是一样的,这就是重叠子问题
我们可以用备忘录来优化


代码如下:

class Solution {
public:
    unordered_map<int, int> mem; // 备忘录
    int numTrees(int n) {
        if (n == 0) {
            return 1;
        }
        if (n == 1) {
            return 1;
        }
        if (mem.find(n) != mem.end()) {
            return mem[n];
        }
        int ans = 0;
        for (int i = 0; i <= n - 1; i++) {
            int l = i;
            int r = n - 1 - i;
            mem[l] = numTrees(l);
            mem[r] = numTrees(r);
            ans += mem[l] * mem[r];
        }
        return ans;
    }
};


2  二叉搜索树


如果二叉树任意节点的值都大于等于左子树所有节点的值且小于等于右子树所有节点的值


则该二叉树称为二叉搜索树 ( Binary Search Tree, BST ), 接下来介绍其常见操作


2.1 判断合法性

给定一棵二叉树,判断是否为二叉搜索树 | leetcode98

还是能用递归解,关键是如何定义子问题,这里提供三种思考的角度


第一是自上而下,即通过递归函数的参数来传递约束,要在前序位置处理父节点

class Solution {
public:
    bool isValidBST(TreeNode* root) {
        return preOrder(root, LONG_MIN, LONG_MAX);
    }
    bool preOrder(TreeNode* root, long minVal, long maxVal) {
        // 边界情况
        if (!root) {
            return true;
        }
        // 基础情况
        // 先检查父节点是否满足约束
        if (
            root -> val <= minVal ||
            root -> val >= maxVal
        ) {
            return false;
        }
        // 再检查两子树是否满足约束
        return (
            preOrder(root -> left , minVal, root -> val) &&
            preOrder(root -> right, root -> val, maxVal)
        );
    }
};


第二是自下而上,即通过递归函数返回值来传递约束,要在后序位置处理父节点

class Solution {
public:
    bool isValidBST(TreeNode* root) {
        pair<long, long> result = postOrder(root);
        return result.first != LONG_MIN && result.second != LONG_MAX;
    }
    // 返回以当前节点为根节点的树中的最小最大值
    pair<long, long> postOrder(TreeNode* root) {
        // 边界情况
        if (!root) {
            return {LONG_MAX, LONG_MIN};
        }
        // 基础情况
        // 先从两子树获取约束
        pair<long, long> left = postOrder(root -> left);
        pair<long, long> right = postOrder(root -> right);
        // 再对父节点检查约束
        long val = root -> val;
        if (
            val <= left.second ||
            val >= right.first
        ) {
            return {LONG_MIN, LONG_MAX};
        }
        // 最后向上层返回当前约束
        return {
            min(val, left.first),
            max(val, right.second)
        };
    }
};


第三是中序遍历,二叉搜索树有个很重要的特性,就是二叉搜索树的中序遍历是所有元素的升序排序

那么我们就可以用数组将二叉搜索树的中序遍历的结果存起来,然后检查数组是否为升序的数组即可

class Solution {
public:
    // 中序遍历结果
    vector<int> elements;
    bool isValidBST(TreeNode* root) {
        inOrder(root);
        // 检查中序遍历结果是否为升序的
        for (int i = 1; i < elements.size(); i++) {
            if (elements[i - 1] >= elements[i]) {
                return false;
            }
        }
        return true;
    }
    // 中序遍历
    void inOrder(TreeNode* root) {
        if (!root) {
            return;
        }
        inOrder(root -> left);
        elements.push_back(root -> val);
        inOrder(root -> right);
    }
};


2.2 查找


在二叉搜索树中查找给定的元素 | leetcode700


这其实就是从上往下遍历,只是根据二叉搜索树的特性能定向遍历,就是每次左右子树只选其一深入

class Solution {
public:
    TreeNode* searchBST(TreeNode* root, int val) {
        // 边界情况
    if (!root) {
            return nullptr;
        }
        // 基础情况
        if (val == root -> val) { // 如果目标值等于当前节点值,则返回结果
            return root;
        }
        else if (val < root -> val) { // 如果目标值小于当前节点值,则去左子树继续找
            return searchBST(root -> left, val);
        }
        else if (val > root -> val) { // 如果目标值大于当前节点值,则去右子树继续找
            return searchBST(root -> right, val);
        }
        return nullptr;
    }
};


2.3 插入


在二叉搜索树中插入给定的元素 | leetcode701


在查找到合适的位置之后做插入,需要注意的是,插入的节点与其父节点要有正确的指向

class Solution {
public:
    TreeNode* insertIntoBST(TreeNode* root, int val) {
        // 边界情况
    if (!root) {
            return new TreeNode(val); // 找到插入位置进行插入,返回插入的节点
        }
        // 基础情况
        // if (val == root -> val) {
        //     不会插入已存在于树中的值
        // }
        if (val < root -> val) { // 如果目标值小于当前节点值,则去左子树找插入位置
            root -> left = insertIntoBST(root -> left, val);
        }
        if (val > root -> val) { // 如果目标值大于当前节点值,则去右子树找插入位置
            root -> right = insertIntoBST(root -> right, val);
        }
        return root;
    }
};


2.4 删除

在二叉搜索树中删除给定的元素 | leetcode450


待删除节点会有三种不同的情况,下面分别说明对三种情况的处理方式

第一,待删除节点的左右孩子都为空,此时直接去删除该节点就好

第二,待删除节点的左右孩子有一个为空,此时让非空的孩子节点接替自己的位置就好

第三,待删除节点的左右孩子都不空,此时情况就复杂了


为了去保持二叉搜索树的性质,需要让左子树中的最大节点或右子树中的最小节点接替自己

找左子树中的最大节点需要从左子树的根节点开始一直往右孩子走直到叶子节点就是目标值为了去保持二叉搜索树的性质,需要让左子树中的最大节点或右子树中的最小节点接替自己

找左子树中的最大节点需要从左子树的根节点开始一直往右孩子走直到叶子节点就是目标值

找右子树中的最小节点需要从右子树的根节点开始一直往左孩子走直到叶子节点就是目标值

class Solution {
public:
    TreeNode* deleteNode(TreeNode* root, int key) {
        // 边界情况
    if (!root) {
            return nullptr;
        }
        // 基础情况
        if (key == root -> val) { // 如果目标值等于于当前节点值,表示已找到待删除元素
            // 处理情况 1、2
            if (root -> left == nullptr) {
                return root -> right;
            }
            if (root -> right == nullptr) {
                return root -> left;
            }
            // 处理情况 3
            // 找到右子树最小节点
            TreeNode* minNode = root -> right;
            while (minNode -> left != nullptr) {
                minNode = minNode -> left;
            }
            // 删除右子树最小节点
            root -> right = deleteNode(root -> right, minNode -> val);
            // 使用右子树最小节点替换当前节点
            minNode -> left = root -> left;
            minNode -> right = root -> right;
            // 返回被删除位置上的新的节点
            root = minNode;
        }
        else if (key < root -> val) { // 如果目标值小于当前节点值,则去左子树删除元素
            root -> left = deleteNode(root -> left, key);
        }
        else if (key > root -> val) { // 如果目标值大于当前节点值,则去右子树删除元素
            root -> right = deleteNode(root -> right, key);
        }
        return root;
    }
};


3  其他


除了二叉树和二叉搜索树之外,还有很多有良好特性的树,这里稍微列举一下,例如


完全二叉树、平衡二叉搜索树、线段树、伸展树、红黑树、前缀树、B 树、B+ 树…


但由于篇幅的原因,这里并不能涵盖所有这些树结构,实在是可惜,以后有机会一定会补上



好啦,本文到此结束,感谢您的阅读!

如果你觉得这篇文章有需要修改完善的地方,欢迎在评论区留下你宝贵的意见或者建议

如果你觉得这篇文章还不错的话,欢迎点赞、收藏、关注,你的支持是对我最大的鼓励 (/ω\)

目录
相关文章
|
2月前
|
存储 算法 搜索推荐
探索常见数据结构:数组、链表、栈、队列、树和图
探索常见数据结构:数组、链表、栈、队列、树和图
112 64
|
12天前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
30 5
|
1月前
|
存储 搜索推荐 算法
【数据结构】树型结构详解 + 堆的实现(c语言)(附源码)
本文介绍了树和二叉树的基本概念及结构,重点讲解了堆这一重要的数据结构。堆是一种特殊的完全二叉树,常用于实现优先队列和高效的排序算法(如堆排序)。文章详细描述了堆的性质、存储方式及其实现方法,包括插入、删除和取堆顶数据等操作的具体实现。通过这些内容,读者可以全面了解堆的原理和应用。
67 16
|
29天前
|
算法
树的遍历算法有哪些?
不同的遍历算法适用于不同的应用场景。深度优先搜索常用于搜索、路径查找等问题;广度优先搜索则在图的最短路径、层次相关的问题中较为常用;而二叉搜索树的遍历在数据排序、查找等方面有重要应用。
27 2
|
2月前
|
存储 算法 关系型数据库
数据结构与算法学习二一:多路查找树、二叉树与B树、2-3树、B+树、B*树。(本章为了解基本知识即可,不做代码学习)
这篇文章主要介绍了多路查找树的基本概念,包括二叉树的局限性、多叉树的优化、B树及其变体(如2-3树、B+树、B*树)的特点和应用,旨在帮助读者理解这些数据结构在文件系统和数据库系统中的重要性和效率。
26 0
数据结构与算法学习二一:多路查找树、二叉树与B树、2-3树、B+树、B*树。(本章为了解基本知识即可,不做代码学习)
|
2月前
【高阶数据结构】二叉树进阶探秘:AVL树的平衡机制与实现详解(三)
【高阶数据结构】二叉树进阶探秘:AVL树的平衡机制与实现详解
|
2月前
【高阶数据结构】二叉树进阶探秘:AVL树的平衡机制与实现详解(二)
【高阶数据结构】二叉树进阶探秘:AVL树的平衡机制与实现详解
|
2月前
|
Java C++
【数据结构】探索红黑树的奥秘:自平衡原理图解及与二叉查找树的比较
本文深入解析红黑树的自平衡原理,介绍其五大原则,并通过图解和代码示例展示其内部机制。同时,对比红黑树与二叉查找树的性能差异,帮助读者更好地理解这两种数据结构的特点和应用场景。
34 0
|
2月前
|
存储 算法
数据结构与算法学习十六:树的知识、二叉树、二叉树的遍历(前序、中序、后序、层次)、二叉树的查找(前序、中序、后序、层次)、二叉树的删除
这篇文章主要介绍了树和二叉树的基础知识,包括树的存储方式、二叉树的定义、遍历方法(前序、中序、后序、层次遍历),以及二叉树的查找和删除操作。
30 0