The problem can be found at the following link: Problem Link
Given a binary tree, the task is to find the maximum path sum. The path may start and end at any node in the tree.
Input:
root[] = [10, 2, 10, 20, 1, N, -25, N, N, N, N, 3, 4]
Output:
42
Explanation:
data:image/s3,"s3://crabby-images/9cea3/9cea3d224a63e9809a79dbd02c9e010c1bfd11c1" alt=""
The maximum path sum is represented using the green colored nodes in the binary tree.
Input:
root[] = [-17, 11, 4, 20, -2, 10]
Output:
31
Explanation:
data:image/s3,"s3://crabby-images/65e64/65e6431e9d65dac2cba0872c479c8ec7d7efe4c8" alt=""
The maximum path sum is represented using the green colored nodes in the binary tree.
- 1 ≤ number of nodes ≤
$10^3$ - -10^4 ≤ node->data ≤
$10^4$
-
DFS Post-Order Traversal:
Use a recursive DFS (post-order) approach to calculate, for each node:- maxSingle: The maximum path sum including the node and at most one of its subtrees.
- Global Maximum: Update a global result with the sum of the node value and the maximum contributions from both left and right subtrees.
-
Steps:
- Recursively compute the maximum path sum from the left and right children.
- Discard negative sums by taking
max(0, value)
. - Update the global maximum with
left + right + node->data
. - Return
node->data + max(left, right)
to allow parent nodes to include the maximum available contribution.
- Expected Time Complexity: O(n), as each node is visited exactly once.
- Expected Auxiliary Space Complexity: O(h), where
h
is the height of the tree (due to the recursion stack). In the worst-case (skewed tree), this can be O(n).
class Solution {
int dfs(Node* r, int& res) {
if (!r) return 0;
int l = max(0, dfs(r->left, res)), rgt = max(0, dfs(r->right, res));
res = max(res, l + rgt + r->data);
return max(l, rgt) + r->data;
}
public:
int findMaxSum(Node* root) {
int res = INT_MIN;
dfs(root, res);
return res;
}
};
- Use post-order traversal to compute two values for each node:
- maxSingle: Maximum path sum including the current node and at most one child.
- maxTop: Maximum path sum where the current node is the highest node (root of the path).
- Recursively compute these values and return the global maximum.
class Solution {
pair<int, int> dfs(Node* r) {
if (!r) return {0, INT_MIN};
auto [lMax, lRes] = dfs(r->left);
auto [rMax, rRes] = dfs(r->right);
int maxSingle = max({r->data, r->data + lMax, r->data + rMax});
int maxTop = max(maxSingle, r->data + lMax + rMax);
return {maxSingle, max({maxTop, lRes, rRes})};
}
public:
int findMaxSum(Node* root) {
return dfs(root).second;
}
};
- Perform an iterative post-order traversal using a stack.
- Maintain a map to store the maximum path sum for each node.
- For each node, compute:
- maxSingle: Max sum including the node and one child.
- maxTop: Max sum through the node with both children.
class Solution {
public:
int findMaxSum(Node* root) {
if (!root) return 0;
stack<Node*> s;
unordered_map<Node*, int> mp;
Node* last = nullptr;
int res = INT_MIN;
while (root || !s.empty()) {
if (root) s.push(root), root = root->left;
else {
Node* node = s.top();
if (node->right && last != node->right) root = node->right;
else {
s.pop();
int l = max(0, mp[node->left]);
int r = max(0, mp[node->right]);
res = max(res, l + r + node->data);
mp[node] = max(l, r) + node->data;
last = node;
}
}
}
return res;
}
};
Approach | Time Complexity | Space Complexity | Pros | Cons |
---|---|---|---|---|
Recursive DFS (Optimized) | 🟢 O(N) | 🟡 O(H) (stack space) | Clean, concise, easy to implement | Stack overflow for deep trees |
Bottom-Up DP | 🟢 O(N) | 🟡 O(H) | Explicit DP states, easy to debug | Slightly verbose |
Iterative Post-Order (Stack) | 🟢 O(N) | 🟡 O(H) | Avoids recursion stack overflow | More complex logic, verbose |
- ✅ Balanced Trees: Recursive DFS is simple and fast.
- ✅ Very Deep Trees: Iterative post-order avoids stack overflow.
- ✅ For Debugging/DP: Bottom-Up DP gives clear intermediate states.
class Solution {
int dfs(Node r, int[] res) {
if (r == null) return 0;
int l = Math.max(0, dfs(r.left, res)), rgt = Math.max(0, dfs(r.right, res));
res[0] = Math.max(res[0], l + rgt + r.data);
return Math.max(l, rgt) + r.data;
}
int findMaxSum(Node root) {
int[] res = {Integer.MIN_VALUE};
dfs(root, res);
return res[0];
}
}
class Solution:
def findMaxSum(self, root):
def dfs(node):
if not node: return 0
l = max(0, dfs(node.left))
r = max(0, dfs(node.right))
nonlocal res
res = max(res, l + r + node.data)
return max(l, r) + node.data
res = float('-inf')
dfs(root)
return res
For discussions, questions, or doubts related to this solution, feel free to connect on LinkedIn: Any Questions. Let’s make this learning journey more collaborative!
⭐ If you find this helpful, please give this repository a star! ⭐