为了账号安全,请及时绑定邮箱和手机立即绑定

leetcode题解(二叉树和递归问题)

标签:
C++ 算法

这篇博客我们主要来看二叉树和二叉树问题中使用递归来解决问题的思路。这类问题有时思路很简单,这是因为二叉树具有天然的递归结构,所以我们使用递归的解法解决二叉树有关的问题就变得非常明显。

<!--more-->

二叉树天然的递归结构
  • 语义
  • 终止条件
  • 递归过程

例子

二叉树前序遍历


//前序遍历node节点
void preorder(TreeNode * node){
    if (node == NULL){ 
        return; //终止条件
    }

    cout << node -> val;
    preorder(node -> left);
    preorder(node -> right);//递归过程
}

二叉树查找某个节点

bool contain(Node *node, key key){
    if (node == NULL)
        return false;

    if (key == node->key)
        return true;

    if( contain(node -> left,key) || contain(node -> right, key))
        return true;

    return false;
}

删除一颗二叉树

void destroy(Node * node){
    if(node == NULL)
        return;
    destroy(node->left);
    destroy(node->right);

    delete node;
    count --;
}

leetcode 104. 二叉树的最大深度

paste

class Solution {
public:
    //求节点root的深度
    int maxDepth(TreeNode* root) {
        //终止条件
        if(root == NULL){ 
            return 0;
        }

        return 1 + max(maxDepth(root -> left), maxDepth(root -> right));

    }
};

相似问题

leetcode 111

一个简单的二叉树问题引发的血案

leetcode 226. 翻转二叉树

paste

代码实现


    /// Definition for a binary tree node.
    struct TreeNode {
        int val;
        TreeNode *left;
        TreeNode *right;
        TreeNode(int x) : val(x), left(NULL), right(NULL) {}
    };

    class Solution {
    public:
        TreeNode* invertTree(TreeNode* root) {

            if ( root == NULL ) {
                return NULL;
            }

            invertTree( root->left );
            invertTree( root->right );
            swap ( root->left, root->right );

            return root;
        }
    };

相似问题

leetcode 100

leetcode 101

leetcode 222

leetcode 110


注意递归的终止条件

有的时候递归的终止条件是存在陷阱的。

leetcode 112. 路径总和

paste

解题思路

采取递归来解决这个问题,我们可以从头结点开始,向它的左右孩子节点寻找sum-根节点的值,不断向下寻找。这道题陷阱在于终止条件不是node == NULL。

/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode(int x) : val(x), left(NULL), right(NULL) {}
 * };
 */
class Solution {
public:
    bool hasPathSum(TreeNode* root, int sum) {
        if (root == NULL){
            return false;
        }

        //为叶子节点
        if (root -> left == NULL && root -> right == NULL){
            return root -> val == sum;
        }

        return hasPathSum(root -> left, sum - root -> val) || hasPathSum(root -> right, sum - root -> val);

        return false;

    }
}; 
定义递归问题

我们如何使用递归的返回值来解决问题。
函数的语义很重要

leetcode 257. 二叉树的所有路径

paste

思路

对于每一个节点,就是该节点的值加上“->”再加上左子树的路径字符串和右子树路径字符串。当到达叶节点时,将字符串加入结果集。

实现

/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode(int x) : val(x), left(NULL), right(NULL) {}
 * };
 */
class Solution {
public:
    vector<string> binaryTreePaths(TreeNode* root) {
        vector<string> res;

        if (root == NULL){
            return res;
        }

        if (root -> left == NULL && root -> right == NULL){
            res.push_back(to_string(root -> val));
            return res;
        }

        vector<string> lefts = binaryTreePaths(root -> left);

        for(int i = 0; i < lefts.size(); i++){
            res.push_back(to_string(root -> val) + "->" + lefts[i]);
        }

        vector<string> rights = binaryTreePaths(root -> right);

        for(int i = 0; i < rights.size(); i++){
            res.push_back(to_string(root -> val) + "->" + rights[i]);
        }

        return res;

    }
};

相似问题

leetcode 113

leetcode 129


稍复杂的递归逻辑

leetcode 437. 路径总和 III

paste

解题思路

猛地一看这个描述,和我们之前的Path Sum是一样的。但是区别在于我们对于路径的定义不一定要起始于根节点,终止于叶子结点,路径可以从任意节点开始,但是只能是向下走的。

paste

在一个节点上要通过三个部分获得答案,第一个部分看看有没有一条路径,它包含node这个节点,并且它的和为sum,这个路径我们进入findPath这个子过程,这个子过程本身又是一个递归函数。另外的两部分就是要在node的左右子树去寻找没有这个ndoe节点的值,有没有这样的路径,他们的和仍然为sum,这件事就是我们PathSum这个函数所做的。在node->left和node->right分别调用PathSum的过程中,他们也会调用findPath来求解。

代码实现

    #include <iostream>

    using namespace std;

    // Definition for a binary tree node.
    struct TreeNode {
        int val;
        TreeNode *left;
        TreeNode *right;
        TreeNode(int x) : val(x), left(NULL), right(NULL) {}
    };

    class Solution {

    public:
        // 在以root为根节点的二叉树中,寻找和为sum的路径,返回这样的路径个数
        int pathSum(TreeNode* root, int sum) {

            if ( root == NULL) {
                return 0;
            }

            int res = findPath( root, sum );
            res += pathSum( root->left,  sum );
            res += pathSum( root->right, sum );

            return res;
        }

    private:

        // 在以node为根节点的二叉树中,寻找包含node的路径,和为sum
        // 返回这样的路径个数
        int findPath( TreeNode* node, int num){

            if ( node == NULL ) {
                return 0;
            }

            int res = 0;
            if ( node->val == num ) {
                res += 1;
            }

            res += findPath( node->left, num - node->val );
            res += findPath( node->right, num - node->val );

            return res;
        }
    };

二分搜索树中的问题

leetcode 235. 二叉搜索树的最近公共祖先

paste

思路

二分搜索树:
每个节点的键值大于左孩子
每个节点的键值小于右孩子
以左右孩子为跟的子树仍为二分搜索树

如果我们给的p,q节点都小于node节点,那么他们最近的公共祖先一定在node左边。如果我们给的p,q节点都大于node节点,那么他们最近的公共祖先一定在ndoe右边。如果一小一大,那么node一定是最近的公众祖先。

代码实现


/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode(int x) : val(x), left(NULL), right(NULL) {}
 * };
 */
class Solution {
public:
    TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {
        assert(p != NULL && q != NULL);

        if (root == NULL){
            return NULL;
        }

        if ((root -> val > p -> val) && (root -> val > q -> val))
            return lowestCommonAncestor(root -> left, p, q);

        if ((root -> val < p -> val) && (root -> val < q -> val))
            return lowestCommonAncestor(root -> right, p, q);

        //p 与q在root的两侧, 则root必为公共祖先,包含p为q的祖先
        return root;
    }
};

相似问题

leetcode 98

leetcode 450

leetcode 108

  • 中序遍历

leetcode 230

leetcode 236


原创始发于慕课网

点击查看更多内容
1人点赞

若觉得本文不错,就分享一下吧!

评论

作者其他优质文章

正在加载中
PHP开发工程师
手记
粉丝
8992
获赞与收藏
336

关注作者,订阅最新文章

阅读免费教程

感谢您的支持,我会继续努力的~
扫码打赏,你说多少就多少
赞赏金额会直接到老师账户
支付方式
打开微信扫一扫,即可进行扫码打赏哦
今天注册有机会得

100积分直接送

付费专栏免费学

大额优惠券免费领

立即参与 放弃机会
意见反馈 帮助中心 APP下载
官方微信

举报

0/150
提交
取消