zoukankan      html  css  js  c++  java
  • LeetCode Path Sum 系列

    题目描述

    LeetCode 三道 Path Sum 的题目,给定一棵二叉树和一个路径和targetSum,分别求问:

    解题思路

    二叉树前序遍历。
    从root到leaf的直接从root开始累计求和即可,采用回溯方式,进分支后累加本节点value,退出分支前减去本节点value;
    任意祖先到子孙结点路径,需要借助从root到当前结点的路径来回溯,到达任意结点后首先判断到root的路径是否符合要求,然后检查这条路径上其他起点的路径是否满足要求;

    参考代码

    Path Sum

    非递归,使用栈作为辅助。

    /*
     * @lc app=leetcode id=112 lang=cpp
     *
     * [112] Path Sum
     */
    
    // @lc code=start
    /**
     * Definition for a binary tree node.
     * struct TreeNode {
     *     int val;
     *     TreeNode *left;
     *     TreeNode *right;
     *     TreeNode() : val(0), left(nullptr), right(nullptr) {}
     *     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
     *     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
     * };
     */
    class Solution {
    public:
        bool hasPathSum(TreeNode* root, int sum) {
            // if(root == nullptr) return (sum == 0);
            if(root == nullptr) return false;
            // BFS或者DFS都可以,这里用DFS可以更快结束
            stack<pair<TreeNode*, int>> stk;
            stk.push({root, root->val});
            while(!stk.empty()) {
                auto p = stk.top();
                stk.pop();
                TreeNode* ptr = p.first;
                int psum = p.second;
                if(psum == sum && ptr->left == nullptr && ptr->right == nullptr) return true;
    
                if(ptr->right) stk.push({ptr->right, psum + ptr->right->val});
                if(ptr->left) stk.push({ptr->left, psum + ptr->left->val});
            }
            return false;
        }
    };
    // @lc code=end
    

    Path Sum II

    回溯,递归解法。

    /*
     * @lc app=leetcode id=113 lang=cpp
     *
     * [113] Path Sum II
     */
    
    // @lc code=start
    /**
     * Definition for a binary tree node.
     * struct TreeNode {
     *     int val;
     *     TreeNode *left;
     *     TreeNode *right;
     *     TreeNode() : val(0), left(nullptr), right(nullptr) {}
     *     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
     *     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
     * };
     */
    class Solution {
    public:
        vector<vector<int>> pathSum(TreeNode* root, int targetSum) {
            if (root == nullptr) return {};
            vector<vector<int>> res;
            vector<int> path;
            find(res, path, root, targetSum);
            return res;
        }
        void find(vector<vector<int>>& res, vector<int>& path, TreeNode* root, int targetSum) {
            assert(root != nullptr);
            path.push_back(root->val);
            targetSum -= root->val;
    
            if (root->left == nullptr && root->right == nullptr) {
                // leaf
                if (targetSum == 0) {
                    res.push_back(path);
                }
                // path.clear();
                // return; // pop_back leaf at end
            }
            if (root->left) find(res, path, root->left, targetSum);
            if (root->right) find(res, path, root->right, targetSum);
    
            path.pop_back();
            targetSum += root->val;
        } // AC
    };
    // @lc code=end
    

    Path Sum III

    回溯,递归解法。

    /*
     * @lc app=leetcode id=437 lang=cpp
     *
     * [437] Path Sum III
     */
    
    // @lc code=start
    /**
     * Definition for a binary tree node.
     * struct TreeNode {
     *     int val;
     *     TreeNode *left;
     *     TreeNode *right;
     *     TreeNode() : val(0), left(nullptr), right(nullptr) {}
     *     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
     *     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
     * };
     */
    class Solution {
    public:
        int pathSum(TreeNode* root, int sum) {
            if (root == nullptr) return {};
            int res = 0;
            vector<int> path;
            find(res, path, root, sum);
            return res;
        }
        void find(int& res, vector<int>& path, TreeNode* root, int targetSum) {
            assert(root != nullptr);
            path.push_back(root->val);
            targetSum -= root->val;
    
            if (targetSum == 0) {
                res++;
            } // path from root to this node
            int tsum = targetSum;
            for (int i=0; i<path.size()-1; i++) {
                if ((tsum += path[i]) == 0) {
                    res++;
                } // path from ancestors to this node
            }
    
            if (root->left) find(res, path, root->left, targetSum);
            if (root->right) find(res, path, root->right, targetSum);
    
            path.pop_back();
            targetSum += root->val;    
        } // AC
    };
    // @lc code=end
    
  • 相关阅读:
    EasyFlash 的初始化配置
    不能靠眼睛之 KEIL 中失效代码灰暗特性
    C 头文件、宏、编译问题
    C++ 中 const、volatile、mutable的用法
    【转】C++ const 关键字总结
    你想要的成都全攻略,好耍不重样——成都胖娃呕心巨作
    【转】RO段、RW段和ZI段 --Image$$??$$Limit 含义(zz)
    深有体会内存对系统性能的重要性
    毕业论文编写笔记
    (二)基于商品属性的相似商品推荐算法——Flink SQL实时计算实现商品的隐式评分
  • 原文地址:https://www.cnblogs.com/zhcpku/p/14352371.html
Copyright © 2011-2022 走看看